iOS 中的 GCD 實現詳解

實現Dispatch Queue須要的軟件組件(代碼)

開發者所使用的GCD的API實現庫地址: libdispatch

如下僅是對Dispatch Queue的分析。數組

GCD隊列是按照層級關係來組織的: bash

GCD中的幾種主要的數據結構:

GCD中的隊列Dispatch Queue是經過鏈表和結構體實現的。網絡

隊列的繼承關係:

GCD中經常使用的結構體和宏

// 聲明一個結構體,能夠看出gcd內部name_t和name_s之間的關係:name_s是一個指向結構體name_t的指針。一個東西
#define DISPATCH_DECL(name) typedef struct name##_s *name##_t
複製代碼
// dispatch_object_t能夠是聯合內部任何一種結構(僅保留Queue相關的)
typedef union {
	struct _os_object_s *_os_obj;// 基類
	struct dispatch_object_s *_do;// 基類繼承os_object
	struct dispatch_continuation_s *_dc;// 任務結構
	struct dispatch_queue_s *_dq;// 隊列結構
	struct dispatch_queue_attr_s *_dqa;// 隊列相關屬性
	struct dispatch_group_s *_dg;// group結構
	struct dispatch_semaphore_s *_dsema;// 信號量
} dispatch_object_t DISPATCH_TRANSPARENT_UNION;
複製代碼
// 系統對象結構或者說系統基類
typedef struct _os_object_s {
	_OS_OBJECT_HEADER(
	const _os_object_vtable_s *os_obj_isa,
	os_obj_ref_cnt,
	os_obj_xref_cnt);
} _os_object_s;

// 系統對象頭部定義宏
#define _OS_OBJECT_HEADER(isa, ref_cnt, xref_cnt) 
        isa;    // isa指針
        int volatile ref_cnt; // gcd對象內部引用計數
        int volatile xref_cnt // gcd對象外部引用計數(內外部都要減到0時,對象會被釋放)
        
複製代碼
struct dispatch_object_s {
	_DISPATCH_OBJECT_HEADER(object);
};
#define _DISPATCH_OBJECT_HEADER(x) 
	struct _os_object_s _as_os_obj[0]; //繼承自os_object
	OS_OBJECT_STRUCT_HEADER(dispatch_##x); 
	struct dispatch_##x##_s *volatile do_next; //鏈表的next
	struct dispatch_queue_s *do_targetq; 
	void *do_ctxt; 
	void *do_finalizer
複製代碼
// GCD中的任務是先被封裝成dispatch_continuation,再提交到隊列中的。該dispatch_continuation用於存儲任務所在的dispatch_group和一些其餘信息,至關於通常常說的執行上下文。
typedef struct dispatch_continuation_s {
	struct dispatch_object_s _as_do[0];//繼承自dispatch_object
	DISPATCH_CONTINUATION_HEADER(continuation);//continuation的一些屬性
} *dispatch_continuation_t;
  
#define DISPATCH_CONTINUATION_HEADER(x) 
	union { 
		const void *do_vtable; 
		uintptr_t dc_flags; 
	}; 
	union { 
		pthread_priority_t dc_priority; 
		int dc_cache_cnt; 
		uintptr_t dc_pad; 
	}; 
	struct voucher_s *dc_voucher; 
	struct dispatch_##x##_s *volatile do_next; 
	dispatch_function_t dc_func; // 任務函數(block會轉爲function)
	void *dc_ctxt; // 執行環境:函數參數
	void *dc_data; 
	void *dc_other
複製代碼
// 隊列結構體
struct dispatch_queue_s {
	_DISPATCH_QUEUE_HEADER(queue);
	DISPATCH_QUEUE_CACHELINE_PADDING; // for static queues only
} DISPATCH_ATOMIC64_ALIGN;

#define _DISPATCH_QUEUE_HEADER(x) 
	struct os_mpsc_queue_s _as_oq[0]; 
	DISPATCH_OBJECT_HEADER(x); //繼承父類的屬性
	uint32_t dq_side_suspend_cnt; 
	DISPATCH_UNION_LE(uint32_t volatile dq_atomic_flags, 
		const uint16_t dq_width, //寬度:串行1,並行>1
		const uint16_t __dq_opaque 
	); 
	DISPATCH_INTROSPECTION_QUEUE_HEADER
	
#define DISPATCH_INTROSPECTION_QUEUE_HEADER 
		TAILQ_ENTRY(dispatch_queue_s) diq_list; 
		dispatch_unfair_lock_s diq_order_top_head_lock; 
		dispatch_unfair_lock_s diq_order_bottom_head_lock; 
		TAILQ_HEAD(, dispatch_queue_order_entry_s) diq_order_top_head; // 隊列鏈表頭結點
		TAILQ_HEAD(, dispatch_queue_order_entry_s) diq_order_bottom_head // 尾節點
  
// 隊列屬性
struct dispatch_queue_attr_s {
	OS_OBJECT_STRUCT_HEADER(dispatch_queue_attr);
};

複製代碼
typedef struct dispatch_continuation_vtable_s {
	_OS_OBJECT_CLASS_HEADER();
	DISPATCH_INVOKABLE_VTABLE_HEADER(dispatch_continuation);
} const *dispatch_continuation_vtable_t;

#define DISPATCH_INVOKABLE_VTABLE_HEADER(x) \
	unsigned long const do_type; // 類型
	const char *const do_kind; // 種類描述
	void (*const do_invoke)(struct x##_s *, dispatch_invoke_context_t, \
			dispatch_invoke_flags_t); // 執行函數
	void (*const do_push)(struct x##_s *, dispatch_object_t, \
			dispatch_qos_t)// 推入隊列函數

串行隊列實例:
DISPATCH_VTABLE_SUBCLASS_INSTANCE(queue_serial, queue,
	.do_type = DISPATCH_QUEUE_SERIAL_TYPE,
	.do_kind = "serial-queue",
	.do_dispose = _dispatch_queue_dispose,
	.do_suspend = _dispatch_queue_suspend,
	.do_resume = _dispatch_queue_resume,
	.do_finalize_activation = _dispatch_queue_finalize_activation,
	.do_push = _dispatch_queue_push,
	.do_invoke = _dispatch_queue_invoke,
	.do_wakeup = _dispatch_queue_wakeup,
	.do_debug = dispatch_queue_debug,
	.do_set_targetq = _dispatch_queue_set_target_queue,
);


複製代碼

GCD中的幾種主要API:

FIFO隊列的管理是經過dispatch_async等函數來實現的(操做頭尾節點)。 在GCD中,可執行的任務有兩種方式實現:Block、Function。所以和任務相關的API通常是有兩種形式的:數據結構

dispatch_async(dispatch_queue_t queue, dispatch_block_t block);
dispatch_async_f(dispatch_queue_t queue,
	void *_Nullable context,
	dispatch_function_t work);
複製代碼

獲取全局隊列

  1. 實現代碼
dispatch_queue_t
dispatch_get_global_queue(long priority, unsigned long flags)
{
        // 形成空值返回的緣由  
	if (flags & ~(unsigned long)DISPATCH_QUEUE_OVERCOMMIT) {
		return DISPATCH_BAD_INPUT;
	} 
        // 轉化優先級
	dispatch_qos_t qos = _dispatch_qos_from_queue_priority(priority);
#if !HAVE_PTHREAD_WORKQUEUE_QOS
	if (qos == QOS_CLASS_MAINTENANCE) {
		qos = DISPATCH_QOS_BACKGROUND;
	} else if (qos == QOS_CLASS_USER_INTERACTIVE) {
		qos = DISPATCH_QOS_USER_INITIATED;
	}
#endif
	if (qos == DISPATCH_QOS_UNSPECIFIED) {
		return DISPATCH_BAD_INPUT;
	}
	// flags傳0時: overcommit = NO
	// 從系統隊列數組中獲取對應優先級的數組
	return _dispatch_get_root_queue(qos, flags & DISPATCH_QUEUE_OVERCOMMIT);
}
複製代碼
static inline dispatch_queue_t
_dispatch_get_root_queue(dispatch_qos_t qos, bool overcommit)
{
	// _dispatch_root_queues是系統中存儲不一樣級別隊列的數組,返回數組中指定index的隊列,overcommit=0/1
	return &_dispatch_root_queues[2 * (qos - 1) + overcommit];
}

// 數組中的隊列
struct dispatch_queue_s _dispatch_root_queues[];
// skip zero
// 1 - main_q  主隊列
// 2 - mgr_q  gcd內部管理隊列
// 3 - mgr_root_q
// 4,5,6,7,8,9,10,11,12,13,14,15 - global queues
// 以後是手動建立的隊列

複製代碼
dispatch_qos_t:
#define DISPATCH_QOS_UNSPECIFIED ((dispatch_qos_t)0)
#define DISPATCH_QOS_MAINTENANCE ((dispatch_qos_t)1)
#define DISPATCH_QOS_BACKGROUND ((dispatch_qos_t)2)
#define DISPATCH_QOS_UTILITY ((dispatch_qos_t)3)
#define DISPATCH_QOS_DEFAULT ((dispatch_qos_t)4)
#define DISPATCH_QOS_USER_INITIATED ((dispatch_qos_t)5)
#define DISPATCH_QOS_USER_INTERACTIVE ((dispatch_qos_t)6)

系統中隊列的優先級枚舉:
enum {
	DISPATCH_ROOT_QUEUE_IDX_MAINTENANCE_QOS = 0,
	DISPATCH_ROOT_QUEUE_IDX_MAINTENANCE_QOS_OVERCOMMIT,
	DISPATCH_ROOT_QUEUE_IDX_BACKGROUND_QOS,
	DISPATCH_ROOT_QUEUE_IDX_BACKGROUND_QOS_OVERCOMMIT,
	DISPATCH_ROOT_QUEUE_IDX_UTILITY_QOS,
	DISPATCH_ROOT_QUEUE_IDX_UTILITY_QOS_OVERCOMMIT,
	DISPATCH_ROOT_QUEUE_IDX_DEFAULT_QOS,
	DISPATCH_ROOT_QUEUE_IDX_DEFAULT_QOS_OVERCOMMIT,
	DISPATCH_ROOT_QUEUE_IDX_USER_INITIATED_QOS,
	DISPATCH_ROOT_QUEUE_IDX_USER_INITIATED_QOS_OVERCOMMIT,
	DISPATCH_ROOT_QUEUE_IDX_USER_INTERACTIVE_QOS,
	DISPATCH_ROOT_QUEUE_IDX_USER_INTERACTIVE_QOS_OVERCOMMIT,
	_DISPATCH_ROOT_QUEUE_IDX_COUNT,
};
複製代碼
  1. 全局隊列的優先級有8中
    除了正常的4種外,還有overcommit的四種
/*!
 * @enum 隊列的一個標誌位
 *
 * @constant DISPATCH_QUEUE_OVERCOMMIT
 * 表示當線程池中線程用完(eg:64個線程),無論系統多麼繁忙,不會等待,這個隊列都會強制建立一個新的線程來執行任務。也就是說能夠建立超過內核數目的線程。
 */
enum {
	DISPATCH_QUEUE_OVERCOMMIT = 0x2ull,
};
複製代碼
  1. 代碼測試:overcommit = YES
    overcommit = NO

主隊列

  1. 實現代碼:
struct dispatch_queue_s _dispatch_main_q = {
	DISPATCH_GLOBAL_OBJECT_HEADER(queue_main),
#if !DISPATCH_USE_RESOLVERS
        // 主隊列的目標隊列是:默認優先級,overcommit的globalQueue
	.do_targetq = &_dispatch_root_queues[
			DISPATCH_ROOT_QUEUE_IDX_DEFAULT_QOS_OVERCOMMIT],
#endif
	.dq_state = DISPATCH_QUEUE_STATE_INIT_VALUE(1) |
			DISPATCH_QUEUE_ROLE_BASE_ANON,
	.dq_label = "com.apple.main-thread",
	.dq_atomic_flags = DQF_THREAD_BOUND | DQF_CANNOT_TRYSYNC | DQF_WIDTH(1),
	.dq_serialnum = 1,
};
複製代碼

建立隊列

  1. 代碼實現
dispatch_queue_t
dispatch_queue_create(const char *label, dispatch_queue_attr_t attr)
{
	return _dispatch_queue_create_with_target(label, attr,
			DISPATCH_TARGET_QUEUE_DEFAULT(NULL), true);
}

static dispatch_queue_t
_dispatch_queue_create_with_target(const char *label, dispatch_queue_attr_t dqa,
		dispatch_queue_t tq, bool legacy)
{
	//
	// Step 1: Normalize arguments (qos, overcommit, tq)
	//

	dispatch_qos_t qos = _dispatch_priority_qos(dqa->dqa_qos_and_relpri);

	_dispatch_queue_attr_overcommit_t overcommit = dqa->dqa_overcommit;
	if (overcommit != _dispatch_queue_attr_overcommit_unspecified && tq) {
		if (tq->do_targetq) {
			DISPATCH_CLIENT_CRASH(tq, "Cannot specify both overcommit and "
					"a non-global target queue");
		}
	}

	if (tq && !tq->do_targetq &&
			tq->do_ref_cnt == DISPATCH_OBJECT_GLOBAL_REFCNT) {
		// Handle discrepancies between attr and target queue, attributes win
		if (overcommit == _dispatch_queue_attr_overcommit_unspecified) {
			if (tq->dq_priority & DISPATCH_PRIORITY_FLAG_OVERCOMMIT) {
				overcommit = _dispatch_queue_attr_overcommit_enabled;
			} else {
				overcommit = _dispatch_queue_attr_overcommit_disabled;
			}
		}
		if (qos == DISPATCH_QOS_UNSPECIFIED) {
			dispatch_qos_t tq_qos = _dispatch_priority_qos(tq->dq_priority);
			tq = _dispatch_get_root_queue(tq_qos,
					overcommit == _dispatch_queue_attr_overcommit_enabled);
		} else {
			tq = NULL;
		}
	} else if (tq && !tq->do_targetq) {
		// target is a pthread or runloop root queue, setting QoS or overcommit
		// is disallowed
		if (overcommit != _dispatch_queue_attr_overcommit_unspecified) {
			DISPATCH_CLIENT_CRASH(tq, "Cannot specify an overcommit attribute "
					"and use this kind of target queue");
		}
	} else {
		if (overcommit == _dispatch_queue_attr_overcommit_unspecified) {
			 // 
			overcommit = dqa->dqa_concurrent ?
					_dispatch_queue_attr_overcommit_disabled :
					_dispatch_queue_attr_overcommit_enabled;
		}
	}
	if (!tq) {
	// 手動建立隊列,未設置目標隊列,則從系統隊列中獲取一個隊列(默認優先級的globalQueue)做爲目標隊列,若是建立的是串行隊列,則目標隊列是overcommit,不然不是(👆)
		tq = _dispatch_get_root_queue(
				qos == DISPATCH_QOS_UNSPECIFIED ? DISPATCH_QOS_DEFAULT : qos,
				overcommit == _dispatch_queue_attr_overcommit_enabled);
	}

	//
	// Step 2: Initialize the queue
	//

	if (legacy) {
		// if any of these attributes is specified, use non legacy classes
		if (dqa->dqa_inactive || dqa->dqa_autorelease_frequency) {
			legacy = false;
		}
	}

	const void *vtable;
	dispatch_queue_flags_t dqf = 0;
	if (legacy) {
		vtable = DISPATCH_VTABLE(queue);
	} else if (dqa->dqa_concurrent) {
		vtable = DISPATCH_VTABLE(queue_concurrent);
	} else {
		vtable = DISPATCH_VTABLE(queue_serial);
	}
	switch (dqa->dqa_autorelease_frequency) {
	case DISPATCH_AUTORELEASE_FREQUENCY_NEVER:
		dqf |= DQF_AUTORELEASE_NEVER;
		break;
	case DISPATCH_AUTORELEASE_FREQUENCY_WORK_ITEM:
		dqf |= DQF_AUTORELEASE_ALWAYS;
		break;
	}
	if (legacy) {
		dqf |= DQF_LEGACY;
	}
	if (label) {
		const char *tmp = _dispatch_strdup_if_mutable(label);
		if (tmp != label) {
			dqf |= DQF_LABEL_NEEDS_FREE;
			label = tmp;
		}
	}
        // 建立隊列對象
        // 分配空間 isa指向它的類型 相關屬性賦值
	dispatch_queue_t dq = _dispatch_object_alloc(vtable,
			sizeof(struct dispatch_queue_s) - DISPATCH_QUEUE_CACHELINE_PAD);
		// 初始化隊列相關參數
	_dispatch_queue_init(dq, dqf, dqa->dqa_concurrent ?
			DISPATCH_QUEUE_WIDTH_MAX : 1, DISPATCH_QUEUE_ROLE_INNER |
			(dqa->dqa_inactive ? DISPATCH_QUEUE_INACTIVE : 0));

	dq->dq_label = label;
	dq->dq_priority = dqa->dqa_qos_and_relpri;
	if (!dq->dq_priority) {
		// 未設置優先級的隊列,優先級默認繼承自它的目標隊列
		_dispatch_queue_priority_inherit_from_target(dq, tq);
	} else if (overcommit == _dispatch_queue_attr_overcommit_enabled) {
		dq->dq_priority |= DISPATCH_PRIORITY_FLAG_OVERCOMMIT;
	}
	if (!dqa->dqa_inactive) {
		_dispatch_queue_inherit_wlh_from_target(dq, tq);
	}
	_dispatch_retain(tq);
	// 設置目標隊列
	dq->do_targetq = tq;
	_dispatch_object_debug(dq, "%s", __func__);
	return _dispatch_introspection_queue_create(dq);
}
// 構建隊列內部頭尾節點等
dispatch_queue_t
_dispatch_introspection_queue_create(dispatch_queue_t dq)
{
	TAILQ_INIT(&dq->diq_order_top_head);// 建立隊列的頭尾節點
	TAILQ_INIT(&dq->diq_order_bottom_head);
	_dispatch_unfair_lock_lock(&_dispatch_introspection.queues_lock);// 加鎖
	TAILQ_INSERT_TAIL(&_dispatch_introspection.queues, dq, diq_list);// 將隊列插入到隊列數組中
	_dispatch_unfair_lock_unlock(&_dispatch_introspection.queues_lock);// 解鎖

	DISPATCH_INTROSPECTION_INTERPOSABLE_HOOK_CALLOUT(queue_create, dq);
	if (DISPATCH_INTROSPECTION_HOOK_ENABLED(queue_create)) {
		_dispatch_introspection_queue_create_hook(dq);
	}
	return dq;
}

複製代碼

設置目標隊列

  1. 實現代碼:
void
_dispatch_queue_set_target_queue(dispatch_queue_t dq, dispatch_queue_t tq)
{
        // global/main queue
	dispatch_assert(dq->do_ref_cnt != DISPATCH_OBJECT_GLOBAL_REFCNT &&
			dq->do_targetq);
        // tq爲空時,設置默認的目標隊列同createQ
	if (unlikely(!tq)) {
		bool is_concurrent_q = (dq->dq_width > 1);
		tq = _dispatch_get_root_queue(DISPATCH_QOS_DEFAULT, !is_concurrent_q);
	}

	if (_dispatch_queue_try_inactive_suspend(dq)) {
		_dispatch_object_set_target_queue_inline(dq, tq);
		return dx_vtable(dq)->do_resume(dq, false);
	}

	if (unlikely(!_dispatch_queue_is_legacy(dq))) {
#if 1
		if (_dispatch_queue_atomic_flags(dq) & DQF_TARGETED) {
			DISPATCH_CLIENT_CRASH(0, "Cannot change the target of a queue "
					"already targeted by other dispatch objects");
		}
#endif
		DISPATCH_CLIENT_CRASH(0, "Cannot change the target of this object "
				"after it has been activated");
	}

	unsigned long type = dx_type(dq);
	switch (type) {
	case DISPATCH_QUEUE_LEGACY_TYPE:
#if 1
		if (_dispatch_queue_atomic_flags(dq) & DQF_TARGETED) {
			_dispatch_bug_deprecated("Changing the target of a queue "
					"already targeted by other dispatch objects");
		}
#endif
		break;
	case DISPATCH_SOURCE_KEVENT_TYPE:
	case DISPATCH_MACH_CHANNEL_TYPE:
		_dispatch_ktrace1(DISPATCH_PERF_post_activate_retarget, dq);
		_dispatch_bug_deprecated("Changing the target of a source "
				"after it has been activated");
		break;
	default:
		DISPATCH_CLIENT_CRASH(type, "Unexpected dispatch object type");
	}

	_dispatch_retain(tq);
	return _dispatch_barrier_trysync_or_async_f(dq, tq,
			_dispatch_queue_legacy_set_target_queue,
			DISPATCH_BARRIER_TRYSYNC_SUSPEND);
}
複製代碼

向隊列中同步添加任務

  1. 實現代碼:
// 任務是block、function走的都是同樣的
void
dispatch_sync(dispatch_queue_t dq, dispatch_block_t work)
{
	dispatch_sync_f(dq, work, _dispatch_Block_invoke(work));
	// _dispatch_Block_invoke:block轉function
}

複製代碼
void
dispatch_sync_f(dispatch_queue_t dq, void *ctxt//函數參數, dispatch_function_t func)
{
// 同步任務通常不切換線程
        // 若是是串行隊列,則等待前面的任務執行完成才能開始執行,barrier經過信號量來實現的
	if (likely(dq->dq_width == 1)) {
		return dispatch_barrier_sync_f(dq, ctxt, func);
	}

	// 當前隊列阻塞barrier或者線程數超過了隊列的容量,可能須要切換線程
	if (unlikely(!_dispatch_queue_try_reserve_sync_width(dq))) {
		return _dispatch_sync_f_slow(dq, ctxt, func, 0);
	}
        // 若是想在當前串行隊列中同步加入任務 死鎖
	_dispatch_introspection_sync_begin(dq);
	if (unlikely(dq->do_targetq->do_targetq)) {
		return _dispatch_sync_recurse(dq, ctxt, func, 0);
	}
	_dispatch_sync_invoke_and_complete(dq, ctxt, func);// 同步執行函數
}
複製代碼

注意死鎖: app

向隊列中異步添加任務

  1. 實現代碼:
dispatch_async(dispatch_queue_t dq, dispatch_block_t work)
{
        // 將任務包裝成continuation
	dispatch_continuation_t dc = _dispatch_continuation_alloc();
	uintptr_t dc_flags = DISPATCH_OBJ_CONSUME_BIT;

	_dispatch_continuation_init(dc, dq, work, 0, 0, dc_flags);
	_dispatch_async_f2(dq, dc);// 中間省略了幾步
}

static void
_dispatch_async_f2(dispatch_queue_t dq, dispatch_continuation_t dc)
{
	if (slowpath(dq->dq_items_tail)) {// 若是當前隊列中尾節點存在,說明當前隊列中有任務了,就不須要再把任務在向上提交了,直接提交到當前隊列中。
		return 	dx_push(dq, dc, _dispatch_continuation_override_qos(dq, dc));
	}

	if (slowpath(!_dispatch_queue_try_acquire_async(dq))) {
		return 	dx_push(dq, dc, _dispatch_continuation_override_qos(dq, dc));
	}
        // 不然須要沿着鏈往上提交任務
	return _dispatch_async_f_redirect(dq, dc,
			_dispatch_continuation_override_qos(dq, dc));
}
// 將任務轉發
_dispatch_async_f_redirect(dispatch_queue_t dq,
		dispatch_object_t dou, dispatch_qos_t qos)
{
	dq = dq->do_targetq;

	// 根據隊列的繼承鏈,將任務一層層向上提交,直到目標隊列是串行隊列,其實就是將任務提交到了目標隊列中了。
	// #define DISPATCH_QUEUE_USES_REDIRECTION(width) \
		({ uint16_t _width = (width); \
		_width > 1 && _width < DISPATCH_QUEUE_WIDTH_POOL; })
		
	while (slowpath(DISPATCH_QUEUE_USES_REDIRECTION(dq->dq_width))) { // 省略中間內容
		dq = dq->do_targetq;
	}

	dx_push(dq, dou, qos);
}

dx_push:

static inline void
_dispatch_queue_push_inline(dispatch_queue_t dq, dispatch_object_t _tail,
		dispatch_qos_t qos)
{
        // 更新隊列尾節點指針,插入新任務節點
	struct dispatch_object_s *tail = _tail._do;
	dispatch_wakeup_flags_t flags = 0;
	bool overriding = _dispatch_queue_need_override_retain(dq, qos);
	if (unlikely(_dispatch_queue_push_update_tail(dq, tail))) {
		if (!overriding) _dispatch_retain_2(dq->_as_os_obj);
		_dispatch_queue_push_update_head(dq, tail);
		flags = DISPATCH_WAKEUP_CONSUME_2 | DISPATCH_WAKEUP_MAKE_DIRTY;
	} else if (overriding) {
		flags = DISPATCH_WAKEUP_CONSUME_2;
	} else {
		return;
	}
	return dx_wakeup(dq, qos, flags);// 喚起下個任務
}

複製代碼

獲取當前隊列

dispatch_get_current_queue()

系統中隊列是按照層級關係來組織的,獲取當前隊列,也就是要獲取執行當前代碼的線程所關聯的隊列,因爲目標隊列的這種繼承關係,任務會一層層向上提交至根隊列,因此dispatch_get_current_queue()獲取的返回結果多是當前執行環境中關聯的隊列中的任意一個與預期不一樣,所以使用它可能會形成錯誤,甚至是死鎖問題。異步

如如下代碼:因爲dispatch_get_current_queue()結果的錯誤會執行else語句,若是當前隊列就是queue,就會形成同步等待(信號量鎖實現,所以也是死鎖問題)async

void func(dispatch_queue_t queue, dispatch_block_t block)
{
    if (dispatch_get_current_queue() == queue) {
        block();
    }else{
        dispatch_sync(queue, block);
    }
}
複製代碼
如何實現獲取當前隊列:

dispatch_queue_set_specific(queue,key) // 給隊列設置一個關聯keyide

dispatch_get_specific(key) // 獲取到當前任務的執行環境(一串繼承鏈的隊列)中對應key值的隊列。這個就更加準確了。函數

dispatch_queue_t q1 = dispatch_queue_create("", NULL);
    dispatch_queue_t q3 = dispatch_queue_create("", NULL);
    dispatch_set_target_queue(q3, q1);

    static int specificKey;
    CFStringRef specificValue = CFSTR("queue1");
    dispatch_queue_set_specific(q1,
                                &specificKey,
                                (void*)specificValue,
                                (dispatch_function_t)CFRelease);
    
    
    dispatch_sync(q3, ^{
        dispatch_block_t block = ^{
            //do something
        };
        // 看看關聯當前這個正在執行的任務的隊列鏈條中是否有key值是specificKey的隊列即q1
        CFStringRef retrievedValue = dispatch_get_specific(&specificKey);
        if (retrievedValue) { // 有,直接執行block
            block();
        } else { // 沒有,同步加到q1中,不會形成同步阻塞
            dispatch_sync(q1, block);
        }
    });
複製代碼

實現代碼:oop

void *
dispatch_get_specific(const void *key)
{
	if (slowpath(!key)) {
		return NULL;
	}
	void *ctxt = NULL;
	dispatch_queue_t dq = _dispatch_queue_get_current();// 獲取當前任務是在哪一個隊列鏈中執行的

	while (slowpath(dq)) {// 沿着鏈找關聯key值的隊列
		ctxt = _dispatch_queue_get_specific_inline(dq, key);
		if (ctxt) break;
		dq = dq->do_targetq;
	}
	return ctxt;
}
複製代碼

可重入概念

GCD使用

通常當下載數據時,爲了避免妨礙主線程的運行,會進行異步處理。那麼每次網絡下載處理都要使用GCD的線程嗎?答案是否認的。若是每次下載就生成一個線程,那麼極可能會產生大量的線程,很快就會用盡線程池中線程和內存。所以系統提供的異步網絡通訊API,它生成一個線程專門用於異步下載,同時開啓這個線程的runloop,添加port讓runloop一直循環運行,這樣線程就不會被銷燬,當有下載任務時,該線程被喚醒執行任務,沒任務就掛起等待。

相關文章
相關標籤/搜索