GCD常用到的方法

1、常用的方法dispatch_async

為了避免界面在處理耗時(shí)的操作時(shí)卡死,比如讀取網(wǎng)絡(luò)數(shù)據(jù),IO,數(shù)據(jù)庫(kù)讀寫(xiě)等,我們會(huì)在另外一個(gè)線程中處理這些操作,然后通知主線程更新界面。
用GCD實(shí)現(xiàn)這個(gè)流程的操作比前面介紹的NSThread NSOperation的方法都要簡(jiǎn)單。代碼框架結(jié)構(gòu)如下:

dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{  
    // 耗時(shí)的操作  
    dispatch_async(dispatch_get_main_queue(), ^{  
        // 更新界面  
    });  
});  

如果這樣還不清晰的話(huà),那我們還是用上兩篇博客中的下載圖片為例子,代碼如下:

dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{  
   NSURL * url = [NSURL URLWithString:@"http://avatar.csdn.net/2/C/D/1_totogo2010.jpg"];  
   NSData * data = [[NSData alloc]initWithContentsOfURL:url];  
   UIImage *image = [[UIImage alloc]initWithData:data];  
   if (data != nil) {  
       dispatch_async(dispatch_get_main_queue(), ^{  
           self.imageView.image = image;  
        });  
   }  
});  

是不是代碼比NSThread NSOperation簡(jiǎn)潔很多,而且GCD會(huì)自動(dòng)根據(jù)任務(wù)在多核處理器上分配資源,優(yōu)化程序。系統(tǒng)給每一個(gè)應(yīng)用程序提供了三個(gè)concurrent dispatch queues。這三個(gè)并發(fā)調(diào)度隊(duì)列是全局的,它們只有優(yōu)先級(jí)的不同。因?yàn)槭侨值?,我們不需要去?chuàng)建。我們只需要通過(guò)使用函數(shù)dispath_get_global_queue去得到隊(duì)列,如下:

dispatch_queue_t globalQ = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);    

這里也用到了系統(tǒng)默認(rèn)就有一個(gè)串行隊(duì)列main_queue
dispatch_queue_t mainQ = dispatch_get_main_queue();    

雖然dispatch queue是引用計(jì)數(shù)的對(duì)象,但是以上兩個(gè)都是全局的隊(duì)列,不用retain或release。

2、dispatch_group_async的使用

dispatch_group_async可以實(shí)現(xiàn)監(jiān)聽(tīng)一組任務(wù)是否完成,完成后得到通知執(zhí)行其他的操作。這個(gè)方法很有用,比如你執(zhí)行三個(gè)下載任務(wù),當(dāng)三個(gè)任務(wù)都下載完成后你才通知界面說(shuō)完成的了。下面是一段例子代碼:

dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);  
dispatch_group_t group = dispatch_group_create();  
dispatch_group_async(group, queue, ^{  
   [NSThread sleepForTimeInterval:1];  
   NSLog(@"group1");  
});  
dispatch_group_async(group, queue, ^{  
   [NSThread sleepForTimeInterval:2];  
   NSLog(@"group2");  
});  
dispatch_group_async(group, queue, ^{  
   [NSThread sleepForTimeInterval:3];  
   NSLog(@"group3");  
});  
dispatch_group_notify(group, dispatch_get_main_queue(), ^{  
   NSLog(@"updateUi");  
});  
dispatch_release(group);  

dispatch_group_async是異步的方法,運(yùn)行后可以看到打印結(jié)果:
gcdTest[43328:11303] group1
gcdTest[43328:12a1b] group2
gcdTest[43328:13003] group3
gcdTest[43328:f803] updateUi
每個(gè)一秒打印一個(gè),當(dāng)?shù)谌齻€(gè)任務(wù)執(zhí)行后,upadteUi被打印。

有a、b、c、d 4個(gè)異步請(qǐng)求,如何判斷a、b、c、d都完成執(zhí)行?如果需要a、b、c、d順序執(zhí)行,該如何實(shí)現(xiàn)?



dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);
dispatch_group_t group = dispatch_group_create();
dispatch_group_async(group, queue, ^{ /*任務(wù)a */ });
dispatch_group_async(group, queue, ^{ /*任務(wù)b */ });
dispatch_group_async(group, queue, ^{ /*任務(wù)c */ }); 
dispatch_group_async(group, queue, ^{ /*任務(wù)d */ }); 
dispatch_group_notify(group,dispatch_get_main_queue(), ^{
 // 在a、b、c、d異步執(zhí)行完成后,會(huì)回調(diào)這里
});
- (void)loadRequest1
{
    dispatch_group_t dispatchGroup = dispatch_group_create();
    dispatch_group_enter(dispatchGroup);
    [MALAFNManger getDataWithUrl:Url1 parameters:nil finish:^(RequestResult result) {
        NSLog(@"第一個(gè)請(qǐng)求完成");
        dispatch_group_leave(dispatchGroup);
    } des:@"第一個(gè)url"];
    dispatch_group_enter(dispatchGroup);
    [MALAFNManger getDataWithUrl:Url2 parameters:nil finish:^(RequestResult result) {
        dispatch_async(dispatch_get_global_queue(0, 0), ^{
            sleep(10);//網(wǎng)絡(luò)請(qǐng)求結(jié)束后回調(diào)是在主線程如果sleep放在外面會(huì)阻塞主線程
            NSLog(@"第二個(gè)請(qǐng)求完成");
            dispatch_group_leave(dispatchGroup);
        });
    } des:@"第二個(gè)url"];
    dispatch_group_notify(dispatchGroup, dispatch_get_main_queue(), ^(){
        NSLog(@"請(qǐng)求完成");
    });
}

3、dispatch_barrier_async的使用

dispatch_barrier_async是在前面的任務(wù)執(zhí)行結(jié)束后它才執(zhí)行,而且它后面的任務(wù)等它執(zhí)行完成之后才會(huì)執(zhí)行
例子代碼如下:

dispatch_queue_t queue = dispatch_queue_create("gcdtest.rongfzh.yc", DISPATCH_QUEUE_CONCURRENT);  
dispatch_async(queue, ^{  
   [NSThread sleepForTimeInterval:2];  
   NSLog(@"dispatch_async1");  
});  
dispatch_async(queue, ^{  
   [NSThread sleepForTimeInterval:4];  
   NSLog(@"dispatch_async2");  
});  
dispatch_barrier_async(queue, ^{  
   NSLog(@"dispatch_barrier_async");  
   [NSThread sleepForTimeInterval:4];  
 
});  
dispatch_async(queue, ^{  
   [NSThread sleepForTimeInterval:1];  
   NSLog(@"dispatch_async3");  
});  

打印結(jié)果:
gcdTest[45547:11203] dispatch_async1
gcdTest[45547:11303] dispatch_async2
gcdTest[45547:11303] dispatch_barrier_async
gcdTest[45547:11303] dispatch_async3
請(qǐng)注意執(zhí)行的時(shí)間,可以看到執(zhí)行的順序如上所述。

4、dispatch_apply

執(zhí)行某個(gè)代碼片段N次。

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

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

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