GCD是一套C寫的多線程模型,根據(jù)隊(duì)列類型和是否同步可以分為并行同步歌憨,并行異步冤竹,串行同步,串行異步乍迄。具體使用可以查看相關(guān)文檔,這里主要介紹為什么主隊(duì)列的同步任務(wù)必然死鎖(當(dāng)然前提需要當(dāng)前線程為主線程后面解釋)而其他串行隊(duì)列中的同步任務(wù)不一定死鎖和用串行隊(duì)列以及柵欄實(shí)現(xiàn)鎖的功能
首先了解一下GCD中同步和異步指代什么:任務(wù)同步指的是阻塞當(dāng)前線程士败,異步不阻塞當(dāng)前線程就乓。往主隊(duì)列中添加一個(gè)同步任務(wù)
eg:
//main thread
dispatch_sync(dispatch_get_main_queue(), ^{
? //do something;
});
同步任務(wù)會阻塞當(dāng)前線程:也就是說main thread將會阻塞,而添加到主隊(duì)列的任務(wù)必然是需要在主線程中執(zhí)行的拱烁,而此時(shí)main thread已經(jīng)被阻塞了生蚁,所以“do something”將不會被執(zhí)行,而主線程又需要等待它執(zhí)行完成戏自,故造成了死鎖現(xiàn)象邦投。
下面再來看看為和向一個(gè)串行隊(duì)列中添加同步任務(wù)不一定死鎖
dispatch_queue_t serialQueeu =dispatch_queue_create("xxxx",DISPATCH_QUEUE_SERIAL);
dispatch_sync(serialQueeu, ^{
//1
dispatch_sync(serialQueeu, ^{
//2
//do something;
? ? });
});
上面代碼不一定會死鎖,
先給結(jié)論:主要原因是gcd的上下文選擇原則擅笔,只有添加到主隊(duì)列的任務(wù)必然是在主線程中執(zhí)行的志衣,也就是自定義串行隊(duì)列中,盡管是在同一個(gè)隊(duì)列中猛们,可能不在同一個(gè)線程中執(zhí)行.?
可以看出1所在的線程被阻塞念脯,如果2處選擇的線程不是1所在的線程,而是一個(gè)新的線程弯淘,則''do smoething"可以在新的線程中執(zhí)行完绿店,block快能正常返回,也就不會一直等待下去,故不會造成死鎖假勿。
NSOperation:
在開發(fā)中我們一般用到的是其兩個(gè)子類:NSInvocationOperation和NSBlockOperation任務(wù)添加分別是以targat-action模式和block借嗽。可以把其加入到一個(gè)由NSOperationQueue創(chuàng)建的隊(duì)列中起飛任務(wù)转培,或者手動調(diào)用start方法恶导。下面主要介紹一下如何自定義NSOperation
在開發(fā)中很多任務(wù)都是異步的,真正任務(wù)完成通過系統(tǒng)回調(diào)得知浸须,時(shí)機(jī)不是由我們決定惨寿。比如再上傳圖片的時(shí)候,我們想嚴(yán)格控制圖片的上傳流程删窒,讓其按照我們指定的順序一張接一張的上傳缤沦。我們想到的可能是按照指定順序把上傳圖片操作加入到一個(gè)并發(fā)量為1的NSOperationQueue中,這只能保證request同步易稠,而結(jié)果則不可控缸废。為了實(shí)現(xiàn)該需求我們可以自定義NSOperation,
@interfaceYYSeiralOperation :NSOperation
+(instancetype)addBlockOperation:(void(^)())operation;
-(void)operationDidFinish;
@end
仿照NSBlockOperation形式提供任務(wù)加載驶社,在start函數(shù)中調(diào)用operation企量,重寫isExecuting isFinished isCancelled,具體細(xì)節(jié)不多說了亡电,提一個(gè)需要注意點(diǎn)届巩,把操作對象加入到隊(duì)列后,有可能因?yàn)橥庖蛭覀冃枰∠尤腙?duì)列中的任務(wù)份乒,[queue cancelAllOperations];執(zhí)行這個(gè)操作的時(shí)候系統(tǒng)會調(diào)用隊(duì)列中所有操作對象的cancle方法恕汇,可能會造成崩潰,當(dāng)一個(gè)操作對象還沒被執(zhí)行或辖,我們設(shè)置操作對象的finish為YES瘾英,則會出錯∷滔荆可以給一個(gè)flag在start函數(shù)執(zhí)行block之前缺谴,把其設(shè)置為YES,在自定義的cancel函數(shù)中判斷這個(gè)標(biāo)志耳鸯,如果存在falg為yes,再設(shè)置finsh相關(guān)屬性湿蛔。
利用GCD實(shí)現(xiàn)鎖的功能
利用GCD的串行隊(duì)列實(shí)現(xiàn)同步,把對一個(gè)對象屬性的讀寫都扔到這個(gè)串行隊(duì)列中县爬,可以使得讀寫的同步(利用串行隊(duì)列FIFO特性)
eg: -(NSString *)someString {
_block NSString *localSomeString;
? ? ? ? dispatch_sync(_syncQueue,^(){
? ? ? ? localSomethinString = _someString;
});
return _someString;(在block中的返回只是退出blcok這個(gè)代碼塊阳啥,不是這個(gè)getter方法)
}
-setSomeString:(NSString *)someString {
dispatch_async(syncQueue,^(){
_someString = someString;
});
}
設(shè)置方法不一定需要同步,讓設(shè)置方法異步執(zhí)行 -setSomeString:(NSString *)someString { dispatch_async(syncQueue,^(){someString = someString; }); } 這種寫法比原來可能還有慢财喳,因?yàn)閳?zhí)行異步派發(fā)時(shí)察迟,是需要拷貝塊。若拷貝快塊時(shí)間明顯超過塊執(zhí)行世界,就會出現(xiàn)比原來慢的請求卷拘。?
使用GCD并發(fā)隊(duì)列來實(shí)現(xiàn)同步鎖 對于屬性的讀寫喊废,我們希望多個(gè)獲取方法可以并發(fā)執(zhí)行祝高,而獲取方法與設(shè)置方法之間不能并發(fā)執(zhí)行栗弟,通過利用GCD的柵欄,可以實(shí)現(xiàn)這個(gè)功能 void dispatch_barrier_async(dispatch_queue_t queue,block); void dispatch_barrier_sync(dispatch_queue_t queue queue,block); dispatch_barrier_async如果傳入自己創(chuàng)建的并行隊(duì)列時(shí)工闺,會阻塞當(dāng)前隊(duì)列執(zhí)行乍赫,而不阻塞當(dāng)前線程。 dispatch_barrier_sync如果傳入自己創(chuàng)建的并行隊(duì)列時(shí)陆蟆,阻塞當(dāng)前隊(duì)列的同時(shí)也會阻塞當(dāng)前線程雷厂,請注意 把寫任務(wù)用dispatch_barrier_async()執(zhí)行,并發(fā)隊(duì)列如果發(fā)現(xiàn)接下來要處理的快是柵欄快,則會一直等待當(dāng)前所有并發(fā)快執(zhí)行完畢叠殷,才會單獨(dú)執(zhí)行這個(gè)柵欄快改鲫,這個(gè)柵欄快執(zhí)行完畢,再按正常方式向下處理林束,所以讀任務(wù)可以通過普通快異步執(zhí)行dispatch_sync();而寫任務(wù)可以通過dispatch_barrier_async這個(gè)柵欄塊加入到并行隊(duì)列中像棘,最后需要注意,這個(gè)隊(duì)列必須是自己創(chuàng)建的不能是全局隊(duì)列壶冒,不然默認(rèn)是dispatch_async()則達(dá)不到同步需求了缕题。
dispatch_semaphore_t介紹
初始化:dispatch_semaphore_t dispatch_semaphore_create(long value); value:大于或等于0,能創(chuàng)建出semaphore對象胖腾,如果小于0則返回一個(gè)NULL對象 value大小等同于一個(gè)線程池的大小烟零,如value為1則允許一個(gè)任務(wù)執(zhí)行,為2則可以同時(shí)允許兩個(gè)任務(wù)執(zhí)行咸作。
相應(yīng)的函數(shù)還有: long dispatch_semaphore_wait(dispatch_semaphore_t dsema, dispatchtimet timeout); 信號量減一锨阿,如果dsema小于0,則阻塞當(dāng)前線程记罚, 返回值是一個(gè)long 類型群井,如果指定了超時(shí)時(shí)間,再等待了指定時(shí)間線程還是沒有被喚醒毫胜,則返回一個(gè)為0书斜,否則返回非0(Returns zero on success, or non-zero if the timeout occurred.); long dispatch_semaphore_signal(dispatch_semaphore_t dsema); 信號量加1酵使,可以用來喚醒被阻塞的線程荐吉,