关于dispatch_sync的死锁

2020-09-07  本文已影响0人  JiangDaDaaa

看源码的时候看到了sync的运用,故将于此有关的死锁按自己的想法理解一下

官方的解析

 Submits a workitem to a dispatch queue like dispatch_async(), however
 dispatch_sync() will not return until the workitem has finished.
 
将工作项提交到dispatch_async()这样的调度队列, 但是,dispatch_sync()在工作项完成之前不会返回。

引用一下 这篇文章的说法:

任务A要想执行完,则必须要等待dispatch_sync函数返回,
然而dispatch_sync要想返回又必须等待任务B完成,
队列又是先进先出的,所以任务B要想执行则必须等待任务A执行完成,
这就形成了相互等待,从而发生死锁.
有些人可能会想凭啥任务B就得排在任务A后面?咋就不能在任务A前面?
我现在就告诉你还真不能,前面已经说了dispatch_sync是不新开线程的,它
现在在当前线程提交一个任务B,那理所当然的任务B就得排在任务A后面.

用代码去看

    // 在主队列上,插入一个同步block块,导致死锁
    dispatch_sync(dispatch_get_main_queue(), ^{
        NSLog(@"do someting");
    });

这里其实看了挺久看不懂的,我们可以这样去想,主线程是同步单线程的.在同步线程上插入了block块,相当于如下代码:

    dispatch_queue_t serialQueue = dispatch_queue_create("com.blbl", DISPATCH_QUEUE_SERIAL);
    
    dispatch_sync(serialQueue, ^{
        NSLog(@"任务A");
        /**
         在当前队列又提交一次同步运行的block,
         导致任务A需要等待任务B返回,而任务A在任务B之前调用,
         所以任务B又需要等待任务A返回了之后才能执行
         */
        dispatch_sync(serialQueue, ^{
            NSLog(@"任务B");
        });
    });

个人比较通俗的理解方式

  1. 由于队列是先进先出的, 所以任务A先运行,所以任务A在队列前方
  2. 任务B这时候同步了一个block块进来,然后任务A就需要去等待任务B完成
  3. 由由于任务A先进来嘛,然后任务B又要等任务A完成
  4. 再通俗一点的说,任务B 插队了 任务A,却又在等任务A完成。😂😂
上一篇下一篇

猜你喜欢

热点阅读