IOS GCD GROUP

一、簡單介紹下將會(huì)用到的一些東西

英語不好就不翻譯官方文檔了..

1、dispatch_group_async

Submits a block to a dispatch queue and associates the block with the givendispatch group

//將一個(gè)block(代碼塊)加入到dispatch_queue_t queue中并和dispatch_group_t group相關(guān)聯(lián)

void

dispatch_group_async(dispatch_group_t group,

dispatch_queue_t queue,

dispatch_block_t block);

個(gè)人理解:將代碼塊dispatch_block_t block放入隊(duì)列dispatch_queue_t queue中執(zhí)行;并和調(diào)度組dispatch_group_t group相互關(guān)聯(lián);如果提交到dispatch_queue_t queue中的block全都執(zhí)行完畢會(huì)調(diào)用dispatch_group_notify并且dispatch_group_wait會(huì)停止等待;

2、dispatch_group_enter(group)、dispatch_group_leave(group)

Calling this function indicates another block has joined the group througha means other than dispatch_group_async(). Calls to this function must be

* balanced with dispatch_group_leave().

調(diào)用這個(gè)方法標(biāo)志著一個(gè)代碼塊被加入了group,和dispatch_group_async功能類似;

需要和dispatch_group_enter()、dispatch_group_leave()成對(duì)出現(xiàn);

void

dispatch_group_enter(dispatch_group_t group);

個(gè)人理解:和內(nèi)存管理的引用計(jì)數(shù)類似,我們可以認(rèn)為group也持有一個(gè)整形變量(只是假設(shè)),當(dāng)調(diào)用enter時(shí)計(jì)數(shù)加1,調(diào)用leave時(shí)計(jì)數(shù)減1,當(dāng)計(jì)數(shù)為0時(shí)會(huì)調(diào)用dispatch_group_notify并且dispatch_group_wait會(huì)停止等待;

3、dispatch_group_notify

void

dispatch_group_notify(dispatch_group_t group,dispatch_queue_t queue,

dispatch_block_t block);

個(gè)人理解:當(dāng)提交到隊(duì)列dispatch_queue_t queue上的所有任務(wù)執(zhí)行完畢時(shí)會(huì)執(zhí)行dispatch_group_notify里的dispatch_block_t block的代碼

4、dispatch_group_wait

long

dispatch_group_wait(dispatch_group_t group, dispatch_time_t timeout);

個(gè)人理解:和dispatch_group_notify功能類似(多了一個(gè)dispatch_time_t參數(shù)可以設(shè)置超時(shí)時(shí)間),在group上任務(wù)完成前,dispatch_group_wait會(huì)阻塞當(dāng)前線程(所以不能放在主線程調(diào)用)一直等待;當(dāng)group上任務(wù)完成,或者等待時(shí)間超過設(shè)置的超時(shí)時(shí)間會(huì)結(jié)束等待;

二、dispatch_group_async代碼示例

- (void)groupSync

{

dispatch_queue_t disqueue =? dispatch_queue_create("com.shidaiyinuo.NetWorkStudy", DISPATCH_QUEUE_CONCURRENT);

dispatch_group_t disgroup = dispatch_group_create();

dispatch_group_async(disgroup, disqueue, ^{

NSLog(@"任務(wù)一完成");

});

dispatch_group_async(disgroup, disqueue, ^{

sleep(8);

NSLog(@"任務(wù)二完成");

});

dispatch_group_notify(disgroup, disqueue, ^{

NSLog(@"dispatch_group_notify 執(zhí)行");

});

dispatch_async(dispatch_get_global_queue(0, 0), ^{

dispatch_group_wait(disgroup, dispatch_time(DISPATCH_TIME_NOW, 5 * NSEC_PER_SEC));

NSLog(@"dispatch_group_wait 結(jié)束");

});

}

向group中放入兩個(gè)任務(wù)(準(zhǔn)確講是將任務(wù)加入到了并行隊(duì)列disqueue中執(zhí)行,然后隊(duì)列和group關(guān)聯(lián)當(dāng)隊(duì)列上任務(wù)執(zhí)行完畢時(shí)group會(huì)進(jìn)行同步),第二個(gè)任務(wù)會(huì)等待8秒所以第一個(gè)任務(wù)會(huì)先完成;會(huì)先打印任務(wù)一完成再打印任務(wù)二完成,當(dāng)兩個(gè)任務(wù)都完成時(shí)dispatch_group_notify中的block會(huì)執(zhí)行;會(huì)接著打印dispatch_group_notify 執(zhí)行;dispatch_group_wait設(shè)置了超時(shí)時(shí)間為5秒所以它會(huì)在5秒后停止等待打印dispatch_group_wait 結(jié)束(任務(wù)二會(huì)等待8秒所以它會(huì)在任務(wù)二完成前打印);

測試輸出結(jié)果

需要注意的:dispatch_group_wait是同步的所以不能放在主線程執(zhí)行。

補(bǔ)充:dispatch_group會(huì)等和它關(guān)聯(lián)的所有的dispatch_queue_t上的任務(wù)都執(zhí)行完畢才會(huì)發(fā)出同步信號(hào)(dispathc_group_notify的代碼塊block會(huì)被執(zhí)行,group_wati會(huì)結(jié)束等待)。也就是說一個(gè)group可以關(guān)聯(lián)多個(gè)任務(wù)隊(duì)列;下面給出示例:

- (void)groupSync2

{

dispatch_queue_t dispatchQueue = dispatch_queue_create("ted.queue.next1", DISPATCH_QUEUE_CONCURRENT);

dispatch_queue_t globalQueue = dispatch_get_global_queue(0, 0);

dispatch_group_t dispatchGroup = dispatch_group_create();

dispatch_group_async(dispatchGroup, dispatchQueue, ^(){

sleep(5);

NSLog(@"任務(wù)一完成");

});

dispatch_group_async(dispatchGroup, dispatchQueue, ^(){

sleep(6);

NSLog(@"任務(wù)二完成");

});

dispatch_group_async(dispatchGroup, globalQueue, ^{

sleep(10);

NSLog(@"任務(wù)三完成");

});

dispatch_group_notify(dispatchGroup, dispatch_get_main_queue(), ^(){

NSLog(@"notify:任務(wù)都完成了");

});

}

上面的代碼里有兩個(gè)隊(duì)列一個(gè)是我自己創(chuàng)建的并行隊(duì)列dispatchQueue,另一個(gè)是系統(tǒng)提供的并行隊(duì)列g(shù)lobalQueue;dispatch_group_notify會(huì)等這兩個(gè)隊(duì)列上的任務(wù)都執(zhí)行完畢才會(huì)執(zhí)行自己的代碼塊。

多個(gè)隊(duì)列執(zhí)行結(jié)果

三、dispatch_group_enter、dispatch_group_level示例

和dispatch_async相比,當(dāng)我們調(diào)用n次dispatch_group_enter后再調(diào)用n次dispatch_group_level時(shí),dispatch_group_notify和dispatch_group_wait會(huì)收到同步信號(hào);這個(gè)特點(diǎn)使得它非常適合處理異步任務(wù)的同步當(dāng)異步任務(wù)開始前調(diào)用dispatch_group_enter異步任務(wù)結(jié)束后調(diào)用dispatch_group_leve

下面是代碼示例:

- (void)groupSync

{

dispatch_group_t group = dispatch_group_create();

dispatch_group_enter(group);

dispatch_async(dispatch_get_global_queue(0, 0), ^{

sleep(5);

NSLog(@"任務(wù)一完成");

dispatch_group_leave(group);

});

dispatch_group_enter(group);

dispatch_async(dispatch_get_global_queue(0, 0), ^{

sleep(8);

NSLog(@"任務(wù)二完成");

dispatch_group_leave(group);

});

dispatch_group_notify(group, dispatch_get_global_queue(0, 0), ^{

NSLog(@"任務(wù)完成");

});

}

示例代碼中在global_queue上執(zhí)行sleep任務(wù)模擬網(wǎng)絡(luò)請求。

控制臺(tái)打印結(jié)果

補(bǔ)充:如果像最后一個(gè)示例那樣,block里執(zhí)行的是同步類型的代碼那么用dispatch_group_async一樣可以達(dá)到同步的效果,但是異步任務(wù)就不行了如下:

- (void)groupSync2

{

dispatch_queue_t dispatchQueue = dispatch_queue_create("ted.queue.next1", DISPATCH_QUEUE_CONCURRENT);

dispatch_queue_t globalQueue = dispatch_get_global_queue(0, 0);

dispatch_group_t dispatchGroup = dispatch_group_create();

dispatch_group_async(dispatchGroup, dispatchQueue, ^(){

dispatch_async(globalQueue, ^{

sleep(5);

NSLog(@"任務(wù)一完成");

});

});

dispatch_group_async(dispatchGroup, dispatchQueue, ^(){

dispatch_async(globalQueue, ^{

sleep(8);

NSLog(@"任務(wù)二完成");

});

});

dispatch_group_notify(dispatchGroup, dispatch_get_main_queue(), ^(){

NSLog(@"notify:任務(wù)都完成了");

});

}

如果dispatch_group_async里執(zhí)行的是異步代碼dispatch_group_notify會(huì)直接觸發(fā)而不會(huì)等待異步任務(wù)完成,而dispatch_group_enter、和dispatch_group_leave則不會(huì)有這個(gè)問題,它們只需要在任務(wù)開始前enter結(jié)束后leave即可達(dá)到線程同步的效果。

引用原作者http://www.itdecent.cn/p/228403206664

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • iOS 多線程系列 -- 基礎(chǔ)概述iOS 多線程系列 -- pthreadiOS 多線程系列 -- NSThrea...
    shannoon閱讀 988評(píng)論 0 2
  • NSThread和NSOperation就不多說,NSOperation就是將GCD封裝成對(duì)象來便于開發(fā)者調(diào)用。蘋...
    Alcander閱讀 615評(píng)論 0 1
  • 背景 擔(dān)心了兩周的我終于輪到去醫(yī)院做胃鏡檢查了!去的時(shí)候我都想好了最壞的可能(胃癌),之前在網(wǎng)上查的癥狀都很相似。...
    Dely閱讀 9,401評(píng)論 21 42
  • 我們知道在iOS開發(fā)中,一共有四種多線程技術(shù):pthread,NSThread,GCD,NSOperation: ...
    請叫我周小帥閱讀 1,559評(píng)論 0 1
  • 一. 重點(diǎn): 1.dispatch_queue_create(生成Dispatch Queue) 2.Main D...
    BestJoker閱讀 1,683評(píng)論 2 2

友情鏈接更多精彩內(nèi)容