iOS 多線程GCD之dispatch_group
本文通過(guò)介紹dispatch_group基本功能,通過(guò)實(shí)例講解dispatch_group的用法。
dispatch_group是GCD(Grand Central Dispatch)中的一組方法,他有一個(gè)組的概念,可以把相關(guān)的任務(wù)歸并到一個(gè)組內(nèi)來(lái)執(zhí)行,通過(guò)監(jiān)聽(tīng)組內(nèi)所有任務(wù)的執(zhí)行情況來(lái)做相應(yīng)處理。
dispatch_group有以下幾種方法
dispatch_group_create
用于創(chuàng)建任務(wù)組
dispatch_group_t dispatch_group_create(void);
dispatch_group_async
把異步任務(wù)提交到指定任務(wù)組和指定下拿出隊(duì)列執(zhí)行
void dispatch_group_async(dispatch_group_t group,
dispatch_queue_t queue,
dispatch_block_t block);
- group ——對(duì)應(yīng)的任務(wù)組,之后可以通過(guò)
dispatch_group_wait或者dispatch_group_notify監(jiān)聽(tīng)任務(wù)組內(nèi)任務(wù)的執(zhí)行情況 - queue ——block任務(wù)執(zhí)行的線程隊(duì)列,任務(wù)組內(nèi)不同任務(wù)的隊(duì)列可以不同
- block —— 執(zhí)行任務(wù)的block
dispatch_group_enter
用于添加對(duì)應(yīng)任務(wù)組中的未執(zhí)行完畢的任務(wù)數(shù),執(zhí)行一次,未執(zhí)行完畢的任務(wù)數(shù)加1,當(dāng)未執(zhí)行完畢任務(wù)數(shù)為0的時(shí)候,才會(huì)使dispatch_group_wait解除阻塞和dispatch_group_notify的block執(zhí)行
void dispatch_group_enter(dispatch_group_t group);
dispatch_group_leave
用于減少任務(wù)組中的未執(zhí)行完畢的任務(wù)數(shù),執(zhí)行一次,未執(zhí)行完畢的任務(wù)數(shù)減1,dispatch_group_enter和dispatch_group_leave要匹配,不然系統(tǒng)會(huì)認(rèn)為group任務(wù)沒(méi)有執(zhí)行完畢
void dispatch_group_leave(dispatch_group_t group);
dispatch_group_wait
等待組任務(wù)完成,會(huì)阻塞當(dāng)前線程,當(dāng)任務(wù)組執(zhí)行完畢時(shí),才會(huì)解除阻塞當(dāng)前線程
long dispatch_group_wait(dispatch_group_t group,
dispatch_time_t timeout);
- group ——需要等待的任務(wù)組
- timeout ——等待的超時(shí)時(shí)間(即等多久),單位為dispatch_time_t。如果設(shè)置為
DISPATCH_TIME_FOREVER,則會(huì)一直等待(阻塞當(dāng)前線程),直到任務(wù)組執(zhí)行完畢
dispatch_group_notify
待任務(wù)組執(zhí)行完畢時(shí)調(diào)用,不會(huì)阻塞當(dāng)前線程
void dispatch_group_notify(dispatch_group_t group,
dispatch_queue_t queue,
dispatch_block_t block);
- group ——需要監(jiān)聽(tīng)的任務(wù)組
- queue ——block任務(wù)執(zhí)行的線程隊(duì)列,和之前group執(zhí)行的線程隊(duì)列無(wú)關(guān)
- block ——任務(wù)組執(zhí)行完畢時(shí)需要執(zhí)行的任務(wù)block
實(shí)例代碼
以下代碼簡(jiǎn)單演示group的使用方法,并測(cè)試group中嵌套異步代碼存在的問(wèn)題
dispatch_group_t group = dispatch_group_create();
dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);
NSLog(@"group one start");
dispatch_group_async(group, queue, ^{
dispatch_async(queue, ^{
sleep(1); //這里線程睡眠1秒鐘,模擬異步請(qǐng)求
NSLog(@"group one finish");
});
});
dispatch_group_notify(group, queue, ^{
NSLog(@"group finished");
});
控制臺(tái)輸出
2016-09-25 09:28:28.716 group one start
2016-09-25 09:28:28.717 group finished
2016-09-25 09:28:29.717 group one finish
從打印結(jié)果可以看出,在group中嵌套了一個(gè)異步任務(wù)時(shí),group并沒(méi)有等待group內(nèi)的異步任務(wù)執(zhí)行完畢才進(jìn)入dispatch_group_notify中,這是因?yàn)?,?code>dispatch_group_async中又啟了一個(gè)異步線程,而異步線程是直接返回的,所以group就認(rèn)為是執(zhí)行完畢了。
對(duì)于以上這種情形,解決方案是使用dispatch_group_enter和dispatch_group_leave方法來(lái)告知group組內(nèi)任務(wù)何時(shí)才是真正的結(jié)束。代碼如下
dispatch_group_t group = dispatch_group_create();
dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);
NSLog(@"group one start");
dispatch_group_enter(group);
dispatch_async(queue, ^{
sleep(1); //這里線程睡眠1秒鐘,模擬異步請(qǐng)求
NSLog(@"group one finish");
dispatch_group_leave(group);
});
dispatch_group_notify(group, queue, ^{
NSLog(@"group finished");
});
控制臺(tái)輸出
2016-09-25 09:34:07.672 group one start
2016-09-25 09:34:08.677 group one finish
2016-09-25 09:34:08.678 group finished
以上代碼,通過(guò)dispatch_group_enter告知group,一個(gè)任務(wù)開(kāi)始,未執(zhí)行完畢任務(wù)數(shù)加1,在異步線程任務(wù)執(zhí)行完畢時(shí),通過(guò)dispatch_group_leave告知group,一個(gè)任務(wù)結(jié)束,未執(zhí)行完畢任務(wù)數(shù)減1,當(dāng)未執(zhí)行完畢任務(wù)數(shù)為0的時(shí)候,這時(shí)group才認(rèn)為組內(nèi)任務(wù)都執(zhí)行完畢了(這個(gè)和GCD的信號(hào)量的機(jī)制有些相似),這時(shí)候才會(huì)回調(diào)dispatch_group_notify中的block。
示例Demo
我做了一個(gè)簡(jiǎn)單的Demo,將網(wǎng)絡(luò)請(qǐng)求的2張圖片拼裝成一張圖片展示。該demo就是使用dispatch_group方法,再兩張圖片都請(qǐng)求完成后,再將其拼裝成一張圖展示。有興趣的童鞋可以看看~