對於GCD中的同步任務、非同步任務的區別,之前有前輩們有這麼認為的:同步任務:不會開闢線程;非同步任務:會開闢新的線程,開闢線程的個數,視資源需求。後來部分人更新了微博簡書稱:同步任務、非同步任務的區別在於前者會阻塞主線程,活著不回阻塞主線程。一開始我也這麼認為,但是後來我寫Demo的時候我就糊塗,不知道...
對於GCD中的同步任務、非同步任務的區別,之前有前輩們有這麼認為的:
- 同步任務:不會開闢線程;
- 非同步任務:會開闢新的線程,開闢線程的個數,視資源需求。
後來部分人更新了微博簡書稱:同步任務、非同步任務的區別在於前者會阻塞主線程,活著不回阻塞主線程。一開始我也這麼認為,但是後來我寫Demo的時候我就糊塗,不知道是我理解錯了,還是怎麼滴,就是解釋不通。下麵是練習代碼:
- (void)testOne { NSLog(@"testOne_start - %@",[NSThread currentThread]); dispatch_queue_t queue = dispatch_queue_create("serial", DISPATCH_QUEUE_SERIAL); dispatch_sync(queue, ^{ NSLog(@"testOne_sync - %@",[NSThread currentThread]); }); NSLog(@"testOne_end - %@",[NSThread currentThread]); }
運行結果如下:
2016-01-15 15:18:09.024 GCD練習[4033:28550] testOne_start - <NSThread: 0x7fafe8506790>{number = 1, name = main}
2016-01-15 15:18:09.025 GCD練習[4033:28550] testOne_sync - <NSThread: 0x7fafe8506790>{number = 1, name = main}
2016-01-15 15:18:09.025 GCD練習[4033:28550] testOne_end - <NSThread: 0x7fafe8506790>{number = 1, name = main}
- (void)testTwo { NSLog(@"testTwo_start - %@",[NSThread currentThread]); dispatch_queue_t queue = dispatch_get_main_queue(); dispatch_sync(queue, ^{ NSLog(@"testTwo_sync - %@",[NSThread currentThread]); }); NSLog(@"testTwo_end - %@",[NSThread currentThread]); }
運行結果如下:testTwo_start - <NSThread: 0x7fc711f046e0>{number = 1, name = main
兩段代碼很簡單,唯一的區別就是前者是把block添加到一個我們新創建的串列隊列中去了,而後者是把block添加到主隊列中去了,而主隊列本身也是一個串列隊列。
按照前人所說,同步任務會阻塞主線程,後者還能解釋的清楚,但是前者我是看了很久都無法解釋。都是主線程被阻塞了,為什麼從自己創建的串列隊列中出列的block任務就能執行,而添加到主隊列中的block任務就不能執行呢? 不明白,那就去查資料唄!這是我查到的一個關於串列隊列的解釋:serial dispatch queue一次只能執行一個任務, 當前任務完成才開始出列並啟動下一個任務。 “當前任務完成才開始出列並啟動下一個任務” 。 所以我在想 是不是同步任務不能立即返回(必須等待block執行完畢),導致下一個任務無法出列執行。簡單的說就是 同步任務是阻塞了隊列,而非阻塞了線程(把同步任務添加到主隊列中,會導致同步任務後面的隊列無法出列執行,跟線程阻塞狀態一樣)。以下是我對上面兩份代碼前者不會導致死鎖以及後者會導致死鎖的個人理解:
- 串列隊列里任務是一個一個執行的,只有當前任務完成才會出列下一任務
在這裡我們創建了一個串列隊列,以及一個同步任務,並將這個同步任務(官方文檔:像隊列提交一個block,並且這個函數不會返回,直到block完成),扔進了這個串列隊列中去了。 程式執行:
1.主隊列 出列第一個NSLog;
2.繼而執行同步任務,同步任務創建了一個任務,並扔進了serial串列隊列中;因為同步任務必須執行完block才返回,故此時主隊列認為 同步任務 尚未完成,因此第三個 NSLog任務無法出列,繼續執行block(從serial出列) ,執行完block同步任務返回。
3.第三個NSLog任務出列。
- 1.主隊列出列 第一個NSLog任務執行完畢。
2.主隊列出列 第二個任務 同步任務, 同步任務創建一個block任務 並添加到主隊列中,以等待出列 執行block,但同步任務 必須等待block執行完畢才能返回,但是對於串列隊列(主隊列)來說只有上一個任務完成了,下一個任務才能出列,所以block任務無法出列 執行,同步任務又等帶block出列執行,死鎖。
上面便是我的理解,也是我自己能對上面兩份代碼作出的唯一解釋,但畢竟跟前人說的相反(在查閱資料的時候也發現有人認為是阻塞隊列,不過被人反駁了),還是希望大神為我指點,以解我心中之疑惑。