iOS-GCD詳解及簡單使用
iOS-GCD 介紹
在開發(fā)過程中,我們有時會希望把一些操作封裝起來延遲一段時間后再執(zhí)行。iOS開發(fā)中,有兩種常用的方法可以實現(xiàn)延遲執(zhí)行,一種是使用GCD,另外一種是使用NSRunLoop類中提供的方法。
前言
對初學(xué)者來說,GCD似乎是一道邁不過去的坎,很多人在同步、異步、串行、并行和死鎖這幾個名詞的漩渦中漸漸放棄治療。本文將使用圖文表并茂的方式給大家形象地解釋其中的原理和規(guī)律。
線程、任務(wù)和隊列的概念
異步、同步 & 并行、串行的特點
一條重要的準(zhǔn)則
一般來說,我們使用GCD的最大目的是在新的線程中同時執(zhí)行多個任務(wù),這意味著我們需要兩項條件:
- 能開啟新的線程
- 任務(wù)可以同時執(zhí)行
- 結(jié)合以上兩個條件,也就等價“開啟新線程的能力 + 任務(wù)同步執(zhí)行的權(quán)利”,只有在滿足能力與權(quán)利這兩個條件的前提下,我們才可以在同時執(zhí)行多個任務(wù)。
所有組合的特點
(一)異步執(zhí)行 + 并行隊列
實現(xiàn)代碼:
//異步執(zhí)行 + 并行隊列 - (void)asyncConcurrent{ //創(chuàng)建一個并行隊列 dispatch_queue_t queue = dispatch_queue_create("標(biāo)識符", DISPATCH_QUEUE_CONCURRENT); NSLog(@"---start---"); //使用異步函數(shù)封裝三個任務(wù) dispatch_async(queue, ^{ NSLog(@"任務(wù)1---%@", [NSThread currentThread]); }); dispatch_async(queue, ^{ NSLog(@"任務(wù)2---%@", [NSThread currentThread]); }); dispatch_async(queue, ^{ NSLog(@"任務(wù)3---%@", [NSThread currentThread]); }); NSLog(@"---end---"); }
打印結(jié)果:
---start---
---end---
任務(wù)3---{number = 5, name = (null)}
任務(wù)2---{number = 4, name = (null)}
任務(wù)1---{number = 3, name = (null)}
解釋:
1.異步執(zhí)行意味著
可以開啟新的線程
任務(wù)可以先繞過不執(zhí)行,回頭再來執(zhí)行
2.并行隊列意味著
任務(wù)之間不需要排隊,且具有同時被執(zhí)行的“權(quán)利”
3.兩者組合后的結(jié)果
開了三個新線程
函數(shù)在執(zhí)行時,先打印了start和end,再回頭執(zhí)行這三個任務(wù)
這三個任務(wù)是同時執(zhí)行的,沒有先后,所以打印結(jié)果是“任務(wù)3-->任務(wù)2-->任務(wù)1”
步驟圖
(二)異步執(zhí)行 + 串行隊列
實現(xiàn)代碼:
//異步執(zhí)行 + 串行隊列 - (void)asyncSerial{ //創(chuàng)建一個串行隊列 dispatch_queue_t queue = dispatch_queue_create("標(biāo)識符", DISPATCH_QUEUE_SERIAL); NSLog(@"---start---"); //使用異步函數(shù)封裝三個任務(wù) dispatch_async(queue, ^{ NSLog(@"任務(wù)1---%@", [NSThread currentThread]); }); dispatch_async(queue, ^{ NSLog(@"任務(wù)2---%@", [NSThread currentThread]); }); dispatch_async(queue, ^{ NSLog(@"任務(wù)3---%@", [NSThread currentThread]); }); NSLog(@"---end---"); }
打印結(jié)果:
---start---
---end---
任務(wù)1---{number = 3, name = (null)}
任務(wù)2---{number = 3, name = (null)}
任務(wù)3---{number = 3, name = (null)}
解釋:
異步執(zhí)行意味著
可以開啟新的線程
任務(wù)可以先繞過不執(zhí)行,回頭再來執(zhí)行
串行隊列意味著
任務(wù)必須按添加進隊列的順序挨個執(zhí)行
兩者組合后的結(jié)果
開了一個新的子線程
函數(shù)在執(zhí)行時,先打印了start和end,再回頭執(zhí)行這三個任務(wù)
這三個任務(wù)是按順序執(zhí)行的,所以打印結(jié)果是“任務(wù)1-->任務(wù)2-->任務(wù)3”
步驟圖
(三)同步執(zhí)行 + 并行隊列
實現(xiàn)代碼:
//同步執(zhí)行 + 并行隊列 - (void)syncConcurrent{ //創(chuàng)建一個并行隊列 dispatch_queue_t queue = dispatch_queue_create("標(biāo)識符", DISPATCH_QUEUE_CONCURRENT); NSLog(@"---start---"); //使用同步函數(shù)封裝三個任務(wù) dispatch_sync(queue, ^{ NSLog(@"任務(wù)1---%@", [NSThread currentThread]); }); dispatch_sync(queue, ^{ NSLog(@"任務(wù)2---%@", [NSThread currentThread]); }); dispatch_sync(queue, ^{ NSLog(@"任務(wù)3---%@", [NSThread currentThread]); }); NSLog(@"---end---"); }
打印結(jié)果:
---start---
任務(wù)1---{number = 1, name = main}
任務(wù)2---{number = 1, name = main}
任務(wù)3---{number = 1, name = main}
---end---
解釋:
同步執(zhí)行執(zhí)行意味著
不能開啟新的線程
任務(wù)創(chuàng)建后必須執(zhí)行完才能往下走
并行隊列意味著
任務(wù)必須按添加進隊列的順序挨個執(zhí)行
兩者組合后的結(jié)果
所有任務(wù)都只能在主線程中執(zhí)行
函數(shù)在執(zhí)行時,必須按照代碼的書寫順序一行一行地執(zhí)行完才能繼續(xù)
注意事項
在這里即便是并行隊列,任務(wù)可以同時執(zhí)行,但是由于只存在一個主線程,所以沒法把任務(wù)分發(fā)到不同的線程去同步處理,其結(jié)果就是只能在主線程里按順序挨個挨個執(zhí)行了
步驟圖
(四)同步執(zhí)行+ 串行隊列
實現(xiàn)代碼:
- (void)syncSerial{ //創(chuàng)建一個串行隊列 dispatch_queue_t queue = dispatch_queue_create("標(biāo)識符", DISPATCH_QUEUE_SERIAL); NSLog(@"---start---"); //使用異步函數(shù)封裝三個任務(wù) dispatch_sync(queue, ^{ NSLog(@"任務(wù)1---%@", [NSThread currentThread]); }); dispatch_sync(queue, ^{ NSLog(@"任務(wù)2---%@", [NSThread currentThread]); }); dispatch_sync(queue, ^{ NSLog(@"任務(wù)3---%@", [NSThread currentThread]); }); NSLog(@"---end---"); }
打印結(jié)果:
---start---
任務(wù)1---{number = 1, name = main}
任務(wù)2---{number = 1, name = main}
任務(wù)3---{number = 1, name = main}
---end---
解釋:
這里的執(zhí)行原理和步驟圖跟“同步執(zhí)行+并發(fā)隊列”是一樣的,只要是同步執(zhí)行就沒法開啟新的線程,所以多個任務(wù)之間也一樣只能按順序來執(zhí)行,
(五)異步執(zhí)行+主隊列
實現(xiàn)代碼:
- (void)asyncMain{ //獲取主隊列 dispatch_queue_t queue = dispatch_get_main_queue(); NSLog(@"---start---"); //使用異步函數(shù)封裝三個任務(wù) dispatch_async(queue, ^{ NSLog(@"任務(wù)1---%@", [NSThread currentThread]); }); dispatch_async(queue, ^{ NSLog(@"任務(wù)2---%@", [NSThread currentThread]); }); dispatch_async(queue, ^{ NSLog(@"任務(wù)3---%@", [NSThread currentThread]); }); NSLog(@"---end---"); }
打印結(jié)果:
---start---
---end---
任務(wù)1---{number = 1, name = main}
任務(wù)2---{number = 1, name = main}
任務(wù)3---{number = 1, name = main}
解釋
異步執(zhí)行意味著
可以開啟新的線程
任務(wù)可以先繞過不執(zhí)行,回頭再來執(zhí)行
主隊列跟串行隊列的區(qū)別
隊列中的任務(wù)一樣要按順序執(zhí)行
主隊列中的任務(wù)必須在主線程中執(zhí)行,不允許在子線程中執(zhí)行
以上條件組合后得出結(jié)果:
所有任務(wù)都可以先跳過,之后再來“按順序”執(zhí)行
步驟圖
(六)同步執(zhí)行+主隊列(死鎖)
實現(xiàn)代碼:
- (void)syncMain{ //獲取主隊列 dispatch_queue_t queue = dispatch_get_main_queue(); NSLog(@"---start---"); //使用同步函數(shù)封裝三個任務(wù) dispatch_sync(queue, ^{ NSLog(@"任務(wù)1---%@", [NSThread currentThread]); }); dispatch_sync(queue, ^{ NSLog(@"任務(wù)2---%@", [NSThread currentThread]); }); dispatch_sync(queue, ^{ NSLog(@"任務(wù)3---%@", [NSThread currentThread]); }); NSLog(@"---end---"); }
打印結(jié)果:
---start---
解釋
- 主隊列中的任務(wù)必須按順序挨個執(zhí)行
- 任務(wù)1要等主線程有空的時候(即主隊列中的所有任務(wù)執(zhí)行完)才能執(zhí)行
- 主線程要執(zhí)行完“打印end”的任務(wù)后才有空
- “任務(wù)1”和“打印end”兩個任務(wù)互相等待,造成死鎖
步驟圖
感謝閱讀,希望能幫助到大家,謝謝大家對本站的支持!
相關(guān)文章
iOS應(yīng)用中使用Auto Layout實現(xiàn)自定義cell及拖動回彈
這篇文章主要介紹了iOS應(yīng)用中使用Auto Layout實現(xiàn)自定義cell及拖動回彈的方法,自定義UITableViewCell并使用Auto Layout對其進行約束可以方便地針對多尺寸屏幕進行調(diào)整,代碼為Swift語言,需要的朋友可以參考下2016-03-03關(guān)于iOS 11下app圖標(biāo)變空白問題的解決方法
升級到iOS11系統(tǒng)下自己的項目桌面app圖標(biāo)不見了,通過查找相關(guān)的資料終于找到了解決方法,下面這篇文章主要給大家介紹了關(guān)于iOS 11下app圖標(biāo)變空白問題的解決方法,文中通過示例代碼介紹的非常詳細(xì),需要的朋友可以參考下。2017-12-12iOS中PNChart與UITableView的聯(lián)動示例詳解
PNChart是個界面很漂亮的圖表第三方庫,UITableView則不用過多介紹了,各位iOS開發(fā)者們都知道,下面這篇文章主要給大家介紹了關(guān)于iOS中PNChart與UITableView的聯(lián)動的相關(guān)資料,需要的朋友可以參考下2018-07-07iOS實現(xiàn)相冊和網(wǎng)絡(luò)圖片的存取
本篇文章主要介紹了iOS實現(xiàn)相冊和網(wǎng)絡(luò)圖片的存取,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2017-04-04iOS定制UISearchBar導(dǎo)航欄同步iOS11的方法
本篇文章主要介紹了iOS定制UISearchBar導(dǎo)航欄同步iOS11的方法,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2017-12-12