iOS多线程之二(GCD)

作者: GitHubPorter | 来源:发表于2016-06-07 14:55 被阅读232次

GCD的基本使用

一、主队列介绍

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

提示:如果把任务放到主队列中进行处理,那么不论处理函数是异步的还是同步的都不会开启新的线程。

获取主队列的方式:

dispatch_queue_t queue=dispatch_get_main_queue();

(1)使用异步函数执行主队列中得任务,代码示例:


执行效果:

(2)使用同步函数,在主线程中执行主队列中得任务,会发生死循环,任务无法往下执行。示意图如下:

二、基本使用

1.问题

任务1和任务2是在主线程执行还是子线程执行,还是单独再开启一个新的线程?


打印结果:

2.开启子线程,加载图片



显示效果:

打印结果:

要求使用GCD的方式,在子线程加载图片完毕后,主线程拿到加载的image刷新UI界面。



打印结果:

好处:子线程中得所有数据都可以直接拿到主线程中使用,更加的方便和直观。

三、线程间通信

从子线程回到主线程


GCD的常见用法

一、延迟执行

1.介绍

iOS常见的延时执行有2种方式

(1)调用NSObject的方法

[self performSelector:@selector(run) withObject:nil afterDelay:2.0];

// 2秒后再调用self的run方法

(2)使用GCD函数

dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(2.0 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{

// 2秒后异步执行这里的代码...

});

2.说明

第一种方法,该方法在那个线程调用,那么run就在哪个线程执行(当前线程),通常是主线程。

[self performSelector:@selector(run) withObject:nil afterDelay:3.0];

说明:在3秒钟之后,执行run函数

代码示例:



说明:如果把该方法放在异步函数中执行,则方法不会被调用(BUG?)

第二种方法,

dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(5.0 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{

//延迟执行的方法

});

说明:在5秒钟之后,执行block中的代码段。

参数说明:

什么时间,执行这个队列中的这个任务。

代码示例:


延迟执行:不需要再写方法,且它还传递了一个队列,我们可以指定并安排其线程。

如果队列是主队列,那么就在主线程执行,如果队列是并发队列,那么会新开启一个线程,在子线程中执行。

二、一次性代码

1.实现一次性代码

需求:点击控制器只有第一次点击的时候才打印。

实现代码:



缺点:这是一个对象方法,如果又创建一个新的控制器,那么打印代码又会执行,因为每个新创建的控制器都有自己的布尔类型,且新创建的默认为NO,因此不能保证改行代码在整个程序中只打印一次。

2.使用dispatch_once一次性代码

使用dispatch_once函数能保证某段代码在程序运行过程中只被执行1次

static dispatch_once_t onceToken;

dispatch_once(&onceToken, ^{

// 只执行1次的代码(这里面默认是线程安全的)

});

整个程序运行过程中,只会执行一次。

代码示例:



效果(程序运行过程中,打印代码只会执行一次):

三、队列组

需求:从网络上下载两张图片,把两张图片合并成一张最终显示在view上。

1.第一种方法

代码示例:



显示效果:

打印查看:

问题:这种方式的效率不高,需要等到图片1.图片2都下载完成后才行。

提示:使用队列组可以让图片1和图片2的下载任务同时进行,且当两个下载任务都完成的时候回到主线程进行显示。

2.使用队列组解决

步骤:

创建一个组

开启一个任务下载图片1

开启一个任务下载图片2

同时执行下载图片1\下载图片2操作

等group中的所有任务都执行完毕, 再回到主线程执行其他操作

代码示例


打印查看(同时开启了两个子线程,分别下载图片):

2.补充说明

有这么1种需求:

首先:分别异步执行2个耗时的操作

其次:等2个异步操作都执行完毕后,再回到主线程执行操作

如果想要快速高效地实现上述需求,可以考虑用队列组

dispatch_group_t group =  dispatch_group_create();

dispatch_group_async(group, dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{

// 执行1个耗时的异步操作

});

dispatch_group_async(group, dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{

// 执行1个耗时的异步操作

});

dispatch_group_notify(group, dispatch_get_main_queue(), ^{

// 等前面的异步操作都执行完毕后,回到主线程...

});

GCD中的死锁

GCD中在主线程中用同步函数分派任务到串行队列中会产生死锁。

互相等待对方完成,举个简单例子好了:

当打印了1以后,主线程调用dispatch_sync这个函数,当这个函数返回的时候主线程才能往下执行。但dispatch_sync返回的条件是里面的Block返回,里面的Block是不会执行的,因为它是被插到主队列最后执行,然而因为dispatch_sync无法返回,所以主队列无法执行到最后一个任务。

NSLog(@"1");

dispatch_sync(dispatch_get_main_queue(), ^{

NSLog(@"2");

});

NSLog(@"3");

还不理解的话再来:

同样用这里来讲解吧,dispatch_sync是调用它的线程上添加一个任务到指定队列,正如这里所示,它接收了一个Block(任务)并放到主队列尾部,即主线程所有的语句都执行完毕后,这个Block才会执行。

但如果队列并不是主队列,而是其他串行或并行,那么系统会创建一条运行在主线程的队列,这条队列并不是主队列,然后Block被加入的是队列的头也是尾,所以先执行并返回,然后dispatch_sync返回。这时对于串行或并行队列其实没有区别,因为dispatch_sync总是在Block返回后才能继续添加下一个任务,不管串行并行,最后都只能一个一个任务按顺序执行。

NSLog(@"1");

dispatch_sync(dispatch_get_main_queue(), ^{

NSLog(@"2");

});

NSLog(@"3");

实在不理解继续···彻底搞懂OC中GCD导致死锁的原因和解决方案:

GCD提供了功能强大的任务和队列控制功能,相比于NSOperationQueue更加底层,因此如果不注意也会导致死锁。

所谓死锁,通常指有两个线程A和B都卡住了,并等待对方完成某些操作。A不能完成是因为它在等待B完成。但B也不能完成,因为它在等待A完成。于是大家都完不成,就导致了死锁(DeadLock)。

有一定GCD使用经验的新手通常认为,死锁是很高端的操作系统层面的问题,离我很远,一般不会遇上。其实这种想法是非常错误的,因为只要简单三行代码(如果愿意,甚至写在一行就可以)就可以人为创造出死锁的情况。

intmain(intargc,constchar* argv[]) {  

@autoreleasepool

{

dispatch_sync(dispatch_get_main_queue(), ^(void){

NSLog(@"这里死锁了");       

});  

}

return0;

}

比如这个最简单的OC命令行程序就会导致死锁,运行后不会看到任何结果。

在解释为什么会死锁之前,首先明确一下“同步&异步”“串行&并发”这两组基本概念:

同步执行:比如这里的dispatch_sync,这个函数会把一个block加入到指定的队列中,而且会一直等到执行完blcok,这个函数才返回。因此在block执行完之前,调用dispatch_sync方法的线程是阻塞的。

与之对应的就有“异步执行”的概念:

异步执行:一般使用dispatch_async,这个函数也会把一个block加入到指定的队列中,但是和同步执行不同的是,这个函数把block加入队列后不等block的执行就立刻返回了。

接下来看一看另一组相对的概念:“串行&并发”

串行队列:比如这里的dispatch_get_main_queue。这个队列中所有任务,一定按照先来后到的顺序执行。不仅如此,还可以保

证在执行某个任务时,在它前面进入队列的所有任务肯定执行完了。对于每一个不同的串行队列,系统会为这个队列建立唯一的线程来执行代码。

与之相对的是并发队列:

并发队列:比如使用dispatch_get_global_queue。这个队列中的任务也是按照先来后到的顺序开始执行,注意是开始,但是它们的执行结束时间是不确定的,取决于每个任务的耗时。对于n个并发队列,GCD不会创建对应的n个线程而是进行适当的优化

我们把整个dispatch_sync看作是一个任务,比如说是非常关键、需要高度集中注意力的运钞过程。这个过程非常重要,一旦开始执行就必须一气呵成,任何事情都不能干扰这个过程(阻塞线程)。

现在主线程开始执行这个运钞任务,任务执行到一半时,突然运钞员说我好累啊,辛苦了好久了,我现在需要休息(向主线程添加了block)。运钞员天真的认为,我知道运钞这个事很重要,本来应该等到运钞结束后再休息(这样是串行)。但是在这之前,我的身体条件不允许工作。

但是之前已经说了,运钞这件事很重要,它一旦开始就不能结束(阻塞线程)。怎么能允许有人中途休息呢,因此要休息可以(block是可以执行的),先把钞票运到安全地方再休息。

对应到代码里面来,当我们想要同步执行这个block的时候,其实是告诉主线程,你把事情处理完了,就过来处理我这个blcok,在此之前我一直等

你。而主线程呢,刚处理dispatch_sync函数到一半呢,这个函数还没返回,哪里有空去执行block。因此这段代码运行后,并非卡在block

中无法返回,而是根本无法执行到这个block。

好了,总结一下,到底什么是死锁。首先,虽然刚刚我们提到了队列和线程,以及它们之间的对应关系,但是死锁一定是针对线程而言的,队列只是GCD给

出的抽象数据结构。所谓的死锁,一定是发生在一个或多个线程之间的。那么死锁和线程阻塞的关系呢,可以这么理解,双向的阻塞导致了死锁。因为阻塞是线程中

经常发生的事情,最多就是主线程的阻塞影响了用户体验。而一旦出现了双向的阻塞,就导致了死锁。我们可以看到,主线程是串行的,在执行某一个任务的时候线

程被阻塞了,而这个任务(dispatch_sync)在执行时,又要求阻塞主线程,从而导致了互相的阻塞,也就是死锁。

接下来我们思考一下,什么情况下会导致死锁。这个问题可能一下子难以得出准确的回答,为了解决这个问题,我打算使用排除法。即先看看什么情况下不会发生死锁。比如说,异步执行block肯定不会发生死锁。比如刚刚的代码改成这样:

dispatch_async(dispatch_get_global_queue(0,0), ^(void){

NSLog(@"这就不死锁了");

});

甚至可以总结出来:异步执行一定不会导致死锁。因为回顾一下之前导致的死锁的原因,很重要的一点是主线程在执行dispatch_sync,这是个

同步方法,block执行完之前都不会返回。而既然是异步的执行,那么是立刻返回的,因此不会阻塞主线程。双向的阻塞不成立了,只是主线程处理blcok

时阻塞,但这不会引起死锁。

根据之前我们的分析和总结,GCD中我们需要关心的就是同步还是异步执行,以及把block添加到哪个队列中(串行还是并发)。

所以接下来就只需要重点思考一下,在同步执行时,什么时候会导致死锁。可以再得出一个结论,向并发队列中添加block不会导致死锁。再次回顾一下

之前导致的死锁的原因,由于在串行队列中添加了block,block一直等到前面的任务处理完才会执行,从而导致了死锁。现在即使是同步的向并发队列中

添加block,GCD会自动为我们管理线程,主线程目前阻塞着(处理这个同步方法),那就新建一个新的线程,但无论如何这个被添加block迟早都会被

执行。而所有添加的block被执行完后,同步方法也就返回了。因此不会导致死锁。

最后再来讨论一下用同步方法向串行队列添加block的情况,这种情况下会不会造成死锁呢,答案是不一定。事实上,导致死锁的原因一定是:

在某一个串行队列中,同步的向这个队列添加block。

比如文章开头的例子就属于这种情况。如果同步的向另外一个串行队列添加方法,并不一定导致死锁。比如:

dispatch_queue_tqueue = dispatch_queue_create("serial",nil);dispatch_sync(queue, ^(void){

NSLog(@"这个也不会死锁");

});

分析一下代码,向名为serial的串行队列添加任务后,GCD自动创建了一个新的线程,在这个线程中执行block方法。在这个过程中,主线程和新的线程都是阻塞的,但是并不会导致死锁。

为什么说向另一个串行队列添加任务不一定导致死锁呢,因为队列是可以嵌套的,比如在A队列(串行)添加一个任务a,在a这个任务中向B队列(串行)

添加任务b,在b这个任务中又向A队列添加任务,这就间接满足了“在某一个串行队列中,同步的向这个队列添加block”。但是我们好像每一次都没有直接

向相同的队列中添加block。

所以判断是否发生死锁的最好方法就是看有没有在串行队列(当然也包括主队列)中向这个队列添加任务。又因为我们知道每个串行队列对应一个线程,所以只要不在某个线程中调用会阻塞这个线程的方法即可。

事实上,我们使用同步的方法编程,往往是要求保证任务之间的执行顺序是完全确定的。且不说GCD提供了很多强大的功能来满足这个需求,向串行队列中

同步的添加任务本身就是不合理的,毕竟队列已经是串行的了,直接异步添加就可以了啊。所以,解决文章开头那个死锁例子的最简单的方法就是在合适的位置添加

一个字母a。

参考博客网站:

http://www.cnblogs.com/wendingding/tag/%E5%A4%9A%E7%BA%BF%E7%A8%8B%E7%AF%87/

http://www.myexception.cn/mobile/2001282.html

http://www.cocoachina.com/bbs/read.php?tid-1482884-page-1.html

相关文章

  • iOS多线程:『GCD』详尽总结

    iOS多线程:『GCD』详尽总结 iOS多线程:『GCD』详尽总结

  • iOS多线程(一)-GCD

    iOS多线程-GCD

  • iOS多线程相关面试题

    iOS多线程demo iOS多线程之--NSThread iOS多线程之--GCD详解 iOS多线程之--NSOp...

  • 多线程之--NSOperation

    iOS多线程demo iOS多线程之--NSThread iOS多线程之--GCD详解 iOS多线程之--NSOp...

  • iOS多线程之--NSThread

    iOS多线程demo iOS多线程之--NSThread iOS多线程之--GCD详解 iOS多线程之--NSOp...

  • [iOS 多线程] iOS多线程-GCD

    iOS多线程-GCD GCD的简介 GCD,全称为 Grand Central Dispatch ,是iOS用来管...

  • iOS 多线程

    iOS 多线程有几种方式 GCD NSOpeartion NSThread phread 多线程 GCD disp...

  • GCD练习

    GCD练习 ios 多线程 GCD : ios 多线程 全剧队列,异步执行 线程间通信 信号量 文件锁 单利模式 ...

  • iOS开发多线程之GCD

    iOS开发多线程之GCDiOS开发之GCD同步任务加强iOS开发之GCD串行队列iOS开发之GCD并发队列 GCD...

  • iOS 多线程

    参考链接 iOS多线程iOS 多线程:『GCD』详尽总结iOS简单优雅的实现复杂情况下的串行需求(各种锁、GCD ...

网友评论

  • _Nevermore:别把文顶顶的代码全部拷贝啊,
    GitHubPorter:@_Nevermore 后面不是备注了出处吗,看了觉得有用mark一下
  • GavinKang:你可以用富文本的代码区写,别贴图片,更好

本文标题:iOS多线程之二(GCD)

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