美文网首页
iOS GCD死锁问题

iOS GCD死锁问题

作者: 丹单当 | 来源:发表于2017-08-07 14:58 被阅读54次

Queue (队列): 分为串行和并行, 串行队列按顺序开始执行,  执行完上一个才能执行下一个. 并行队列(开始也是按添加顺序执行,不需要等其他的完成) 允许同时执行多个任务, 完成的顺序是随机的

Async和Sync (异步执行和同步执行): 使用dispathc_async调用一个block, 这个block会被放到队尾执行,  至于block是串行执行还是并行执行只和dispatch_async中的参数里面指定的queue是并行的还是串行的有关,  但是dispatch_async会马上返回.

而dispatch_sync同样也是把block放到指定的queue上面执行, 但是会等待这个block执行完毕才会返回, 阻塞当前的queue直到sync返回

1. 所以, 当前queue是串行队列的时候


dispatch_sync(dispatch_get_main_queue(), ^{

NSLog(@"2");

});


这段代码如果在主线程执行的, 那么就会造成死锁,  因为dispatch_sync函数指定的线程和函数所在的都是主线程(dispatch_get_main_queue), 而dispatch_sync要dispatch_get_main_queue执行完dispatch_sync中的block才会有返回,  而dispatch_get_main_queue中又在执行dispatch_sync这个函数, 要等到dispatch_sync函数返回才回去执行block. 这就造成了dispatch_sync永远都无法返回了(死锁).

总结:  主要原因就在于dispatch_sync所在的线程和函数指定的线程是同一个线程,且是串行的.  而串行队列执行原则是一个完成在执行下一个的. 这就好比, 我们去某地买房,  但是买房需要先有户口,  而想要户口却必须要有房子. 这就造成了你一辈子都没办法在这地方买房了.  类比可能稍微有些不恰当, 但是大体是这个意思. 理解就好

会阻塞当前线程(在这种情况是主线程), 而主线程被阻塞之后,  因为dispatch_sync把block放到了主线程的队尾, 因此, 主线程需要等dispatch_sync函数返回之后才会继续执行下去, 也就是去执行block,  而dispatch_sync 需要block执行完毕才返回.  因此,  这个时候就会造成死锁问题,

于此类似的


dispatch_queue_t queue1 = dispatch_queue_create("com.baizhong.queue1", DISPATCH_QUEUE_SERIAL);  //串行

dispatch_async(queue1, ^{

NSLog(@"1");

dispatch_sync(queue1, ^{

NSLog(@"2");

});

});

NSLog(@"3");


上面这段代码也会死锁

因为dispatch_sync函数指定的线程和函数所在的线程都是queue1,  而queue1 是串行队列

下面就不会了


dispatch_queue_t queue = dispatch_queue_create("com.baizhong.queue1", DISPATCH_QUEUE_SERIAL);

dispatch_async(queue, ^{

NSLog(@"1");

dispatch_sync(dispatch_get_main_queue(), ^{

NSLog(@"2");

});

});

NSLog(@"3");


因为dispatch_async所在的当前线程和dispatch_async函数中指定的线程不是同一个

2. 当前的queue是并行队列的时候

第一种情况:


dispatch_queue_t queue2 = dispatch_queue_create("com.baizhong.queue1", DISPATCH_QUEUE_CONCURRENT); //并行

dispatch_async(queue2, ^{

NSLog(@"1");

dispatch_sync(queue2, ^{

NSLog(@"这种不会死锁");

});

});

NSLog(@"3");


这种情况因为queue是并行队列,  所以 NSLog(@"2")会马上执行,所以dispatch_sync函数也不会一直等待不返回造成死锁

第二种情况:


dispatch_queue_t queue = dispatch_queue_create("com.baizhong.queue1", DISPATCH_QUEUE_CONCURRENT);

dispatch_async(queue, ^{

NSLog(@"1");

dispatch_sync(dispatch_get_main_queue(), ^{

NSLog(@"这种不会死锁");

});

});

NSLog(@"3");


这种也是一样

总结起来就是:

造成死锁的主要原因就是,在某一个串行队列中, 同步的向这个队列添加block.

上面列举的只是一部分情况, 还有其他的各种情况可以自己动手写写, 但是归根结底原因都是差不多

关于队列与线程

当我们创建了队列之后,我们需要把任务添加到队列中,并指定以同步还是异步的方式执行添加到队列中的任务。

   同步,其会在当前线程立即执行添加的任务(无论是串行还是并行队列都如此)。

   异步,其会新创建一个新的线程来执行任务。而异步对于串行和并行队列的又不一样的意义的。

对于异步执行的串行队列的话,新添加的多个任务会在新创建的线程中依次执行,即一个执行完在执行另一个任务,有点类似同步执行的样子(其区别可以看做是同步不会创建新的线程,而异步会创建新的线程,且只创建一个)。

如果是并发队列的话,使用同步方式执行任何则和串行队列一样。而使用异步方式执行任务的话,新添加的任务都会放到新创建的线程,即每个任务在单独线程中,并且所有的任务都是并发执行的,而不像串行队列是有顺序的。如果是异步方式执行的话,则每个任务都会新开一个线程,并且这些任务是并发执行的。

参考链接

GCD死锁问题

串行队列和并发队列

欢迎指正!

相关文章

  • 死锁 GCD 多线程

    死锁 GCD 多线程 Ios - LDSmallCat - 博客园 Ios中GCD死锁困扰很多人,分享一点个人经...

  • iOS GCD死锁问题

    Queue (队列): 分为串行和并行, 串行队列按顺序开始执行, 执行完上一个才能执行下一个. 并行队列(开始也...

  • IOS开发 GCD产生死锁的总结

    在IOS开发中GCD的使用频率很高,但是使用不当,则会产生死锁,以下是我的对GCD产生死锁的总结。 了解死锁之前首...

  • iOS gcd线程死锁问题

    同步异步决定是否具备开启线程的能力串行并行决定代码执行的先后顺序 先看下这几个场景,每个场景中的代码执行后会打印什...

  • 多线程相关之GCD、死锁、dispatch_barrier_as

    推荐阅读:备战2020——iOS全新面试题总结 GCD---同步/异步 ,串行/并发 死锁 GCD任务执行顺序 d...

  • ios GCD死锁

    死锁 1.定义 所谓死锁,通常指的是两个线程T1和T2都被卡住,并等待对方完成某些操作,T1等待T2完成,T2等待...

  • iOS:GCD死锁

    出现死锁的特点: 用 sync函数,向 当前 串行队列 添加任务。 举例:

  • 2019 iOS面试题-----进程、线程、多进程、多线程、任务

    2019 iOS面试题-----多线程相关之GCD、死锁、dispatch_barrier_async、dispa...

  • [iOS] GCD死锁问题解读

    一.解题 今天在群里发现了一道比较有意思的题, 仔细想了一下, 还是值得深思的, 所以写出来记录一下, 以下均为本...

  • iOS-知识点

    五个案例让你明白GCD死锁:http://ios.jobbole.com/82622/ iOS开发中多线程间关于锁...

网友评论

      本文标题:iOS GCD死锁问题

      本文链接:https://www.haomeiwen.com/subject/cbcalxtx.html