GCD线程死锁

作者: 迷路的安然和无恙 | 来源:发表于2017-07-06 12:29 被阅读75次

先看👇一段代码

- (void)viewDidLoad
{
    [super viewDidLoad];
    NSLog(@"as_one");
    dispatch_sync(dispatch_get_main_queue(), ^{
        NSLog(@"as_two");
    });
    NSLog(@"as_three");
}

有什么问题?
这段程序应该怎么执行?
是one-two-three还是one-three-two?

线程死锁

你会发现,程序在输出as_one后就挂了,这就引出了线程死锁的问题。
原理如下:GCD的队列可以理解为三种类型

  • The main queue :主队列,主线程就是在个队列中。
  • Global queues : 全局并发队列。
  • 用户队列:是用函数 dispatch_queue_create 创建的自定义队列
    dispatch_sync 和 dispatch_async 区别:

dispatch_async(queue,block) async 异步队列,dispatch_async` 函数会立即返回, block会在后台异步执行。

dispatch_sync(queue,block) sync 同步队列,dispatch_sync 函数不会立即返回,及阻塞当前线程,等待 block同步执行完成。

分析上面的代码

如下:

viewDidLoad 在主线程中, 及在dispatch_get_main_queue() 中,执行到sync 时 向dispatch_get_main_queue()插入同步 threed1.

sync 会等到后面block 执行完成才返回, sync 又再dispatch_get_main_queue() 队列中,它是串行队列,sync 是后加入的,前一个是主线程,所以sync想执行block必须等待主线程执行完成,主线程等待sync返回,去执行后续内容。

死锁就这样产生了,sync 等待mainThread执行完成,mianThread等待sync 函数返回。

解决方法

- (void)viewDidLoad
{
    [super viewDidLoad];

    dispatch_async(dispatch_get_global_queue(0, 0), ^{
    
        NSLog(@"as_one");
    
        dispatch_sync(dispatch_get_main_queue(), ^{
            NSLog(@"as_two");

        });
       NSLog(@"as_three");
   });
}

程序会完成执行,执行结果

//  控制台输出
as_one
as_two
as_three。

为什么?

首先: async在主线程中创建了一个异步线程加入全局并发队列,async不会等待block执行完成,立即返回。
1.async立即返回,viewDidLoad执行完毕,及主线程执行完毕。
2.同时,全局并发队列立即执行异步block, 打印 1, 当执行到 sync 它会等待 block 执行完成才返回, 及等待dispatch_get_main_queue() 队列中的 mianThread 执行完成, 然后才开始调用block

因为1 和 2 几乎同时执行,因为2 在全局并发队列上, 2 中执行到sync 时 1 可能已经执行完成或 等了一会,mainThread 很快退出, 2 等已执行后续内容。

证明

👆说因为主线程及时退出,程序才会执行as_two。怎么证明呢。

- (void)viewDidLoad {
    [super viewDidLoad];
    dispatch_async(dispatch_get_global_queue(0, 0), ^{
        NSLog(@"as_one");
        dispatch_sync(dispatch_get_main_queue(), ^{
            NSLog(@"as_two");
        });
        NSLog(@"as_three");
    });
    NSLog(@"开始阻塞主线程");
//  来个一直执行的大循环
    while (1) {}
    NSLog(@"结束阻塞主线程");
}
//  控制台输出
开始阻塞主线程
as_one

结语

这个时候你看下那个几乎100%的CPU就知道主线程,他累坏了.....

相关文章

  • 带你分分钟了解GCD多线程的死锁

    带你了解GCD多线程的死锁 阐述: 1.什么是GCD? GCD,全称 Grand Central Dispatch...

  • 死锁 GCD 多线程

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

  • GCD避免死锁的三要素

    平时总在用GCD,但你知不知道,GCD一不小心就会出现死锁,如果死锁在主线程上,整个程序就完了,所以避免死锁是我们...

  • GCD线程死锁

    GCD 以下情况下会死锁,(不考虑线程锁的情况下) 主队列中同步添加任务。 串行队列任务中添加同步任务 为什么呢?...

  • GCD线程死锁

    GCD简化了多线程的使用,但使用不当会导致线程死锁。 dispatch_async(queue,block) a...

  • GCD线程死锁

    先看?一段代码 有什么问题?这段程序应该怎么执行?是one-two-three还是one-three-two? 线...

  • GCD线程死锁

    dispatch_sync是同步线程,若当前线程运行在主线程里面,那就是主线程。然后,在主线程执行log。此时,d...

  • iOS多线程(一)

    多线程涉及到的概念: 进程,线程,主线程,任务,队列,死锁,串行,并行,同步,异步,GCD,NSOperation...

  • GCD死锁

    GCD死锁原因 GCD死锁的原因是队列阻塞,而不是线程阻塞! 串行和并行 串行和并行都是相对于队列而言的-队列(负...

  • iOS基础篇-GCD死锁

    GCD死锁概念: 使用GCD多线程编程时,如果同步函数dispatch_sync(),目标queue为当前queu...

网友评论

    本文标题:GCD线程死锁

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