在查看GCD的官方API文檔時,其中的一句話深深的深深地打動了我:
原文鏈接:
https://developer.apple.com/reference/dispatch
翻譯過來大致意思就是:
嘗試在主隊列上同步執行任務會導致死-鎖.
官方文檔就這樣給你一個結論,沒有多余的解釋(不跟你多BB).
通常在非主線程中執行完耗時的任務后,需要向主隊列中提交任務來實現與主線程的通信,在主線程中進行UI的更新.但是我使用dispatch_sync()函數嘗試向主隊列提交同步執行的任務并沒有導致死鎖.
經過一番思索和測試,對于"使用dispatch_sync()函數嘗試向主隊列提交同步執行的任務是否會導致"死鎖"這個問題,我終于能夠給自己一個說的過去的解釋了.
結合代碼闡述一下我的理解:
<pre>
<code>
`
// ViewController.m
// dead-lock 死鎖
import "ViewController.h"
@interface ViewController ()
@end
@implementation ViewController
(IBAction)willNotResultInDeadLock
{
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^
{
/源文件第15行/NSLog(@"%d,currentThread->%@",LINE,[NSThread currentThread]);
/源文件第16行/dispatch_sync(dispatch_get_main_queue(), ^
/源文件第17行/{
/源文件第18行/NSLog(@"%d,currentThread->%@",LINE,[NSThread currentThread]);
/源文件第19行/});
});
/*
這個block(第15~19行代碼)在非主線程中執行,當CPU調度到這個block所在的線程時,執行這個block.
與此同時主線程不被CPU調度,主線程此時不能執行任何任務,所以向主隊列中提交同步執行的任務(第18行代碼)不會造成死鎖.
當CPU重新調度到主線程時,這個被提交的同步執行的任務(第18行代碼)會與在主隊列排在它之前的其他任務按照FIFO的順序出隊并在主線程中執行.
*/
}-
(IBAction)willResultInDeadLock
{
dispatch_sync(dispatch_get_main_queue(), ^
{
NSLog(@"%d,currentThread->%@",LINE,[NSThread currentThread]);
});
/*
當-willResultInDeadLock方法被調用時,-willResultInDeadLock方法整體是一個任務將在主線程執行.假如-willResultInDeadLock方法能夠成功返回,被提交到主隊列中的同步任務(NSLog()函數)就能出隊并且會在主線程中執行.
然而主隊列是串行FIFO隊列,前一個任務在主線程完成后下一個任務才能夠出隊并在主線程中執行.
必須在dispatch_sync()函數返回后,-willResultInDeadLock方法才能返回.
必須在NSLog()函數返回后,dispatch_sync()函數才能返回.
NSLog()函數作為主隊列中的任務,必須等待主線程中的任務完成后(-willResultInDeadLock方法返回后)才能從主隊列中出隊并在主線程中執行.所以三者之間產生死鎖如下:
dispatch_sync()函數----<-------wait for----------- -willResultInDeadLock方法
| |
| /|\\
| |
| |
|--------wait for----->---NSLog()函數--------wait for--------|
*/
}
- (void)viewDidLoad
{
[super viewDidLoad];
}
@end
`
</code>
</pre>
調用- (IBAction)willNotResultInDeadLock方法,輸出如下:
2016-12-29 17:38:21.949 dead-lock 死鎖[23295:163231] 15,currentThread-><NSThread: 0x600000079b00>{number = 3, name = (null)} 2016-12-29 17:38:21.951 dead-lock 死鎖[23295:163146] 18,currentThread-><NSThread: 0x600000068280>{number = 1, name = main}
調用- (IBAction)willResultInDeadLock方法導致了死鎖,程序崩潰:
"知其然而不知其所以然"的感覺常常讓我在編程時候很不安,總要擔心程序會不會有潛在的問題,遇到很抽象的難理解的問題時,能解決的話我通常會花幾個小時,花幾天來解決.實在是解決不了,就放到備忘錄中,留著以后有空的時候解決.你們是否也有這種不安的感覺?