美文网首页selector
GCD中主队列和全局队列

GCD中主队列和全局队列

作者: wind_dy | 来源:发表于2017-06-01 16:01 被阅读73次

(1)主队列

主队列是GCD自带的一种特殊的串行队列,放在主队列中得任务,都会放到主线程中执行。

dispatch_get_main_queue()

下面主要研究一下在主队列中的同步和异步问题:

<1>同步

- (void)viewDidLoad {

[superviewDidLoad];

NSLog(@"%@",[NSThreadcurrentThread]);

dispatch_sync(dispatch_get_main_queue(), ^{

NSLog(@"sync----%@",[NSThreadcurrentThread]);

});

NSLog(@"%@",[NSThreadcurrentThread]);

}

打印结果:

你可能会奇怪:为什么只有一条打印记录?其实这里出现了死锁,所以后面的打印,包括同步任务里面的打印都无法执行,为什么会发生死锁呢,关键问题出现在设置的dispatch_sync方法里面的第一个参数,下面来分析一下。

首先,系统执行时,肯定是从主线程执行第一个打印,之后执行到dispatch_sync方法,这是同步方法,这个方法第一个参数是dispatch_get_main_queue(),所以会将block的打印任务放到主队列中等待执行。此时,主线程阻塞在这,等待同步任务的执行。但是,主队列串行队列,一个任务执行结束才能取出下一个任务执行,而此时当前的任务被阻塞了,无法结束,所以后一个任务(即刚刚添加的同步任务)必须等待前一个任务结束后才能执行。此时,主线程等待着同步任务执行结束返回,而同步任务等待着主队列的前一个任务(即当前被阻塞的任务)执行结束后再执行。所以出现了死锁的现象。我们应该尽量避免这种情况的发生。那么如果换成dispatch_async方法,是否就不会出现死锁了呢?

<2>异步

- (void)viewDidLoad {

[superviewDidLoad];

NSLog(@"%@",[NSThreadcurrentThread]);

dispatch_async(dispatch_get_main_queue(), ^{

NSLog(@"async----%@",[NSThreadcurrentThread]);

});

NSLog(@"%@",[NSThreadcurrentThread]);

}

打印结果:

显然,这种情况下,所有的打印都能完成,简单分析一下:主线程首先执行第一个打印,执行到dispatch_async方法时,往主队列添加了一个异步任务,此时并不阻塞主线程,主线程立刻返回执行后面的打印。主线程当前的任务完成了,然后从主队列取出下一个任务(刚刚添加的异步任务)来执行,完成打印async—-。所以不会出现死锁现象。

(2)全局队列

dispatch_get_global_queue(longidentifier,unsignedlongflags)

第一个参数identifier: 在ios7中代表队列优先级(priority),在ios8及以后代表服务质量(A quality of service),一般设置为DISPATCH_QUEUE_PRIORITY_DEFAULT即可。

两者的对应关系如下:

DISPATCH_QUEUE_PRIORITY_HIGH: QOS_CLASS_USER_INITIATED

DISPATCH_QUEUE_PRIORITY_DEFAULT: QOS_CLASS_DEFAULT

DISPATCH_QUEUE_PRIORITY_LOW: QOS_CLASS_UTILITY

DISPATCH_QUEUE_PRIORITY_BACKGROUND: QOS_CLASS_BACKGROUND

第二个参数flags:保留参数,以便以后使用,一般传0即可。

在全局队列中,同步任务是否也会发生死锁现象呢?

<1>同步

- (void)viewDidLoad {

[superviewDidLoad];

NSLog(@"%@",[NSThreadcurrentThread]);

dispatch_sync(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT,0), ^{

NSLog(@"sync----%@",[NSThreadcurrentThread]);

});

NSLog(@"%@",[NSThreadcurrentThread]);

}

打印结果:

从打印结果来看,并没有死锁现象,仔细分析一下:主线程先执行第一句打印,然后执行dispatch_sync方法,往全局队列中添加一个同步任务,此时主线程应该阻塞住,等待全局队列中同步任务的执行完毕,而全局队列dispatch_sync方法并不会创建新线程,那肯定需要在主线程执行(上一篇的结论),可是主线程已经阻塞了,所以按理说此时应该发生死锁,可是为什么从打印结果看它确实是在主线程执行的同步任务,而且并没有发生死锁现象呢?

对应这个问题,我利用前面和上面的知识都无法解释,我在网上查了很多,都没有提到过这个问题,难道是我上面的分析有问题,按照上面的分析,那所有的同步方法,无论是自定义队列还是全局队列、主队列,同步任务都会出现死锁现象。难道我钻了牛角尖。没有找到中文的解释,我去官方文档是查,也一无所获,最后我无意的点进了dispatch_sync的源代码中,在该方法上面的注释中发现了这么一句话:As an optimization, dispatch_sync() invokes the block on the current thread when possible.大致意思是:苹果系统内部做了优化,将所有的dispatch_sync()方法中block的执行都放在当前的线程中,在我们这里也就是主线程中。这也解释了上一篇留下的一个问题:为什么所有的同步方法都在主线程中执行 ?可是,这个解释,好像并不能回答为什么这里没有发生死锁的问题?我也一直没有找到合理的解释,如果哪位有比较合理的解释,麻烦给我留言。

我个人的猜测:是否会在需要执行dispatch_sync()方法时,将当前任务先挂起,只要能取出要执行的任务来就让主线程先执行,之后再来执行挂起任务。这个猜测似乎和主队列同步方法发生死锁的解释不矛盾,主队列中的任务只能前一个执行完后一个才能取出来执行,所以在主队列中后面的同步任务无法取出来,自然会发生死锁。而全局队列中,同步任务就是最前面的任务,所以能取出来,所以可以由主线程来执行,执行后主线程可以返回执行后面的操作。

转载自feisongfs

相关文章

  • GCD的队列类型与NSOperationQueue的队列类型

    GCD的队列类型 并发队列自己创建的全局 串行队列主队列自己创建的 NSOperationQueue的队列类型 主...

  • iOS--GCD四大队列

    队列是FIFO,先进先出原则 原则上,GCD只有两种队列,串行队列和并发队列。全局队列是系统提供的一个并发队列,主...

  • 2021--- GCD

    gcd同步,异步,串行队列,并发队列,全局队列,主队列,以及死锁。 1、gcd队列阻塞问题[https://www...

  • iOS开发笔记-多线程的使用方法

    多线程方式一:GCD队列的3个种类: 自建队列: 分并行/串行 全局队列: 属于并行队列, 是系统默认创建的. 主...

  • GCD串行并发队列

    学习完本篇,您会对以下知识点更加理解: 队列 串行队列 并发队列 GCD全局队列 GCD主队列 创建串行队列 创建...

  • GCD相关知识总结

    一:GCD的队列简介 GCD的Queue有三大队列类型:主队列(main)、全局队列(global)、用户队列(c...

  • GCD中主队列和全局队列

    (1)主队列 主队列是GCD自带的一种特殊的串行队列,放在主队列中得任务,都会放到主线程中执行。 dispatch...

  • iOS - GCD

    GCD队列 串行队列 并行队列 全局队列 主队列 几个容易混淆的概念dispatch_barrier_async栅...

  • GCD多线程问题整理

    1.GCD队列有哪几种类型?有哪几种队列? GCD队列分为串行队列、并行队列两种类型;队列有主串行队列、全局并行队...

  • iOS 网络开发

    多线程 NSThread GCD队列并发队列全局队列自己创建串行队列自己创建主队列任务:block函数sync:同...

网友评论

    本文标题:GCD中主队列和全局队列

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