美文网首页
iOS线程锁死的思考

iOS线程锁死的思考

作者: 青鸟evergreen | 来源:发表于2018-05-07 17:21 被阅读42次
      dispatch_sync(dispatch_get_main_queue(), ^{
            NSLog(@"2");
        });
      NSLog(@"3");
    
    

    对于上述代码造成锁死原因,要厘清几个概念:Queue 和 Async、Sync。
    Queue分串行,并行,见名知意。
    Async:异步执行,使用dispatch_async 调用一个block,这个block会被放到指定的queue队尾等待执行,但是dispatch_async会马上返回。
    Sync:同步执行,使用dispatch_async 调用一个block,这个block同样会被放到指定的queue队尾等待执行,并且阻塞当前queue直到sync函数返回。

    name针对上述问题就好理解了,当前队列为Main_queue, 同时指定的队列也是main_queue。block添加到main_queue队尾执行,但是因为此处为同步串行队列,所以阻塞了当前线程,那么block任务无法执行,sync函数也就无法返回,也就造成了线程阻塞,锁死。

    dispatch_queue_t queue = dispatch_queue_create("abc",             
        DISPATCH_QUEUE_SERIAL);
        dispatch_sync(queue, ^{
            NSLog(@"%@", [NSThread currentThread]);
     });
     NSLog(@"1");
    

    此处同样是同步串行队列为什么没造成死锁呢?
    因为阻塞的队列和指定的队列不是同一个队列

    NSLog(@"1"); // 任务1
    dispatch_sync(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_HIGH, 0), ^{
        NSLog(@"2"); // 任务2
    });
    NSLog(@"3"); // 任务3
    
    //顺序 1、2、3
    

    首先执行任务1,接下来会遇到一个同步线程,程序会进入等待。等待任务2执行完成以后,才能继续执行任务3。从dispatch_get_global_queue可以看出,任务2被加入到了全局的并行队列中,当并行队列执行完任务2以后,返回到主队列,继续执行任务3。

    dispatch_queue_t queue = dispatch_queue_create("com.demo.serialQueue", DISPATCH_QUEUE_SERIAL);
    NSLog(@"1"); // 任务1
    dispatch_async(queue, ^{
        NSLog(@"2"); // 任务2
        dispatch_sync(queue, ^{  
            NSLog(@"3"); // 任务3
        });
        NSLog(@"4"); // 任务4
    });
    NSLog(@"5"); // 任务5
    
    //顺序:1、2、5 (2,5顺序不一定)
    

    这个案例没有使用系统提供的串行或并行队列,而是自己通过dispatch_queue_create函数创建了一个DISPATCH_QUEUE_SERIAL的串行队列。

    执行任务1;
    遇到异步线程,将【任务2、同步线程、任务4】加入串行队列中。因为是异步线程,所以在主线程中的任务5不必等待异步线程中的所有任务完成;
    因为任务5不必等待,所以2和5的输出顺序不能确定;
    任务2执行完以后,遇到同步线程,这时,将任务3加入串行队列;
    又因为任务4比任务3早加入串行队列,所以,任务3要等待任务4完成以后,才能执行。但是任务3所在的同步线程会阻塞,所以任务4必须等任务3执行完以后再执行。这就又陷入了无限的等待中,造成死锁。

     dispatch_async(dispatch_get_global_queue(0, 0), ^{
             NSLog(@"1"); // 任务1
            dispatch_sync(dispatch_get_main_queue(), ^{
                NSLog(@"2"); // 任务2
            });
            NSLog(@"3"); // 任务3
     });
    NSLog(@"4"); // 任务4
    while (1) {
    }
    NSLog(@"5");//任务5
    
    //顺序1、4,顺序不一定
    

    和上面几个案例的分析类似,先来看看都有哪些任务加入了Main Queue:【异步线程、任务4、死循环、任务5】。
    在加入到Global Queue异步线程中的任务有:【任务1、同步线程、任务3】。

    第一个就是异步线程,任务4不用等待,所以结果任务1和任务4顺序不一定。
    任务4完成后,程序进入死循环,Main Queue阻塞。但是加入到Global Queue的异步线程不受影响,继续执行任务1后面的同步线程。

    同步线程中,将任务2加入到了主线程,并且,任务3等待任务2完成以后才能执行。这时的主线程,已经被死循环阻塞了。所以任务2无法执行,当然任务3也无法执行,在死循环后的任务5也不会执行。

    最终,只能得到1和4顺序不定的结果

    相关文章

      网友评论

          本文标题:iOS线程锁死的思考

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