dispatch_sync到底为什么死锁

之前看面试题知道下面代码会死锁,网上说是因为相互等待,等等,总觉得难以理解

dispatch_sync(dispatch_get_main_queue(), ^{

NSLog(@"死锁了");

});

经过一下午的仔细研究,算是有点收获,下面分享给大家,不对的地方欢迎指正


一:首先介绍下基本概念

队列:用于存放任务。一共有两种队列,串行队列并行队列

同步(sync)和异步(async)的主要区别在于会不会阻塞当前线程,直到Block中的任务执行完毕!

如果是同步(sync)操作,它会阻塞当前线程并等待Block中的任务执行完毕,然后当前线程才会继续往下运行。

如果是异步(async)操作,当前线程会直接往下执行,它不会阻塞当前线程


二:死锁

官方文档指出:dispatch_sync的当前执行队列与提交block执行的目标队列相同时将造成死锁。

Submits a block to a dispatch queue for synchronous execution. Unlike dispatch_async, this function does not return until the block has finished. Calling this function and targeting the current queue results in deadlock.

dispatch_sync(dispatch_get_main_queue(), ^{

NSLog(@"死锁了");

});


根据文档所说,此任务本身(整个代码)在主队列中,而要添加的任务(NSLog(@"死锁了");)也要添加到主队列,它会死锁。这是为什么呢?

假设自己为A任务,A任务所在的队列为主队列,A任务正在执行,A任务的代码是添加B任务到主队列并使B任务执行完毕,然后B任务就被添加到主队列,可是它要等待A任务执行完,才轮到它执行,A任务等待B任务,B任务又等待A任务,因此形成死锁。

为了验证观点,我试了下面的代码:

dispatch_queue_t queue =dispatch_queue_create("queue",DISPATCH_QUEUE_SERIAL);

dispatch_sync(queue, ^{

       NSLog(@"正常打印");

      dispatch_sync(queue, ^{

                 NSLog(@"死锁了");

                });

     });

NSLog(@"死锁了");

因为是同步往“queue”添加任务,可是本身又是在“queue”中执行,再次循环等待,死锁发生。

参考自链接:

https://www.zhihu.com/question/23338200/answer/57844956

http://www.jianshu.com/p/0b0d9b1f1f19

你可能感兴趣的:(dispatch_sync到底为什么死锁)