iOS 关于多线程及死锁的正确解答

很多人弄不懂这四个名词,网上也有很多的误导。 对于线程死锁更是解释的五花八门、煞有介事的样子。

一.引出此文的元凶---网上的曲解

这里讲解的煞有其事啊,忽悠的我一愣一愣的,任务3阻挡了任务2的执行,那么我不写任务3的话是否就不死锁了呢?

经过我的代码验证,事实是只写:

//当前队列为主队列
dispatch_sync(dispatch_main_queue(),^ {
NSLog(@"");
});
复制代码

也会造成死锁,说明什么任务1,2,3都是骗人的,那到底是什么造成的思死锁呢???请认真看文章。

----- 装逼开始,咳咳,讲解开始 -----

二.揭开背后的真相

####1.先要知道概念 这里就真正的解释下他们的意思,我们先抛出概念,再列举例子,再根据例子理解概念: 异步:sync 同步:async 队列:queue 串行:serial 并行(并发):concurrent

queue:队列分为串行和并行,队列是任务的容器。就像排队买东西,串行是大家排成一队一个一个的买,先来的先买,后来的后买;并行是大家并排排,同时买,谁先买完的看脸。

串行、并行队列

同步、异步: 使用dispatch_async:调用一个block,这个block会被放到指定的queue队尾等待执行,至于这个block是并行还是串行只和dispatch_async参数里面指定的queue是并行还是串行有关。但是当前队列会直接跳过block,也就是不去管block的情况,dispatch_async直接执行完毕

使用dispatch_sync :dispatch_sync 方法会被加入当前队列,而且dispatch_sync 会等待block执行完毕才return,block被放到指定的queue上面执行,block里的代码执行完(即代码执行到block结束的}),这时候整个dispatch_sync才算执行完说白了就是dispatch_sync正在出队列,但是要等block执行完才能完全出队列。

下面看代码:


2.死锁的犯人就是 --- 他自己

死锁示例

控制台输出

死锁原因图解

死锁发生在 当前队列为串行队列,并通过dispatch_sync往当前队列添加了task的时候。可以看出,死锁的是队列,而不是线程。代码中,第一个dispatch_sync并没有造成死锁,就是因为block在另外的一个队列里,而线程还是可以执行这个队列的任务的,执行完之后,队列就畅通了


2.示例讲解各种组合情况

- (void)test2 {
    NSLog(@"主线程");
    dispatch_queue_t concurrent = dispatch_queue_create("test.euque.concurrent", DISPATCH_QUEUE_CONCURRENT);
    dispatch_queue_t serial = dispatch_queue_create("test.queue.serial", DISPATCH_QUEUE_SERIAL);
    //async,主线程不会等待block的完成,会直接执行gcd之后的代码:NSLog(@"主线程任务结束")
    //task进入concurrent并行队列,由于是async所以允许concurrent开辟新线程
    dispatch_async(concurrent, ^{
        NSLog(@"concurrent_thread");
        //1.再次开辟新线程运行:1秒后NSLog(@"1");
        dispatch_async(concurrent, ^{
            [NSThread sleepForTimeInterval:1];
            NSLog(@"1");
        });
        //2.由于是sync,dispatch_sync在concurrent队列中等待block执行结束
//block被加入concurrent,由于是并行,dispatch_sync在队列里等待的时候并不耽误其他函数出队列,所以block依旧可以出队列执行,所以不会死锁。
//  sync说明block在concurrent_thread线程中执行
        dispatch_sync(concurrent, ^{
            [NSThread sleepForTimeInterval:2];
            NSLog(@"2");
        });
        //3. NSLog(@"2")运行结束后,async,concurrent_thread不必等待block的执行,直接前往4
        //NSLog(@"3")被加入serial队列,出队列时async开辟新线程输出3
        dispatch_async(serial, ^{
            NSLog(@"3");
        });
        //4. dispatch_sync在concurrent中等待block的执行,block添加到serial队列,sync说明block交给concurrent_thread运行,输出4
        dispatch_sync(serial, ^{
            NSLog(@"4");
        });
    });
    NSLog(@"主线程任务结束");
}

复制代码

执行结果


3.不同于自定义队列的主队列

//如果执行这个函数的队列就是main_queue,则会死锁
 dispatch_sync(dispatch_get_main_queue(), ^{
        
    });

 dispatch_async(dispatch_get_main_queue(), ^{
        
    });
复制代码

对于主队列来说,无论是sync还是async都不会开辟新线程,因为主队列的任务只在main_thread执行,那么这两个函数的区别就是是否需要在当前队列中等待block执行完毕。

简书地址:www.jianshu.com/p/af68aab13…

猜你喜欢

转载自juejin.im/post/5b4829a95188251b1c3ce66b