iOS之GCD

在iOS中实现多线程的方案中,GCD是一种很好的方案。GCD在工作时会自动利用更多的处理器核心,以充分利用更强大的机器。GCD是Grand Central Dispatch的简称,它是基于C语言的。如果使用GCD,完全由系统管理线程,我们不需要编写线程代码。只需定义想要执行的任务,然后添加到适当的调度队列(dispatch queue)。GCD会负责创建线程和调度你的任务,系统直接提供线程管理。
这里就需要提及GCD的两个核心概念了,一个是任务另一个是队列了,除此之外还有四个重要的名词术语:同步、异步、串行和并发。
简单的说,任务就是执行的操作(也就是想要做的事情);队列就是用来存放任务的容器。同步就是在当前线程中执行任务,需要的是马上执行任务;异步就是在另一个线程中执行任务;串行就是任务一个一个按顺序的执行;并发就是多个任务同时执行。

1. GCD的基本使用方法

首先要明确GCD的使用步骤大致有三步:

  1. 创建队列
  2. 确定任务
  3. 将任务放到队列中去执行,这里GCD会自动的将任务从队列中取出来放到相应的线程中去执行。

1.1串行同步队列

a.创建队列

dispatch_queue_t queue = dispatch_queue_create("同步串行", DISPATCH_QUEUE_SERIAL);

这里的dispatch_queue_t相当于一个基本的数据类型,这里指的是队列类型。
第二个参数:DISPATCH_QUEUE_SERIAL也就是串行,而并发是DISPATCH_QUEUE_CONCURRENT
b.确定任务并且放到队列中

dispatch_sync(queue, ^{
        // 这里的代码就是要执行的任务
        NSLog(@"%@",[NSThread currentThread]);
    });

dispatch_sync表示的同步。
打印日志如下:

<NSThread: 0x7f93d2d05ee0>{number = 1, name = main}

c.将多个任务放到队列中
这里就可以多创建几个任务。

for (int i = 0; i < 5; ++i) {
        dispatch_sync(queue, ^{
            //
            [self operation:i];
        });
  }
 NSLog(@"同步串行over");

下面这是控制台的打印日志,

<NSThread: 0x7f93d2d05ee0>{number = 1, name = main} ~ 0 <NSThread: 0x7f93d2d05ee0>{number = 1, name = main} ~ 1
<NSThread: 0x7f93d2d05ee0>{number = 1, name = main} ~ 2
<NSThread: 0x7f93d2d05ee0>{number = 1, name = main} ~ 3
<NSThread: 0x7f93d2d05ee0>{number = 1, name = main} ~ 4
同步串行over

从上面的打印日志可以看出,任务始终都是在同一个线程中进行的操作。
所以对于同步串行而言,在主线程中任务是一个一个按顺序执行;先执行任务,再执行“同步串行over”

1.2串行异步

串行异步,首先异步一定会在新的线程中执行,所以任务会在新开辟的子线程中一个个的执行任务,爱奇艺的缓存也就是这么做的。
a.创建串行队列

dispatch_queue_t queue = dispatch_queue_create("串行异步", NULL);

上面讲到DISPATCH_QUEUE_SERIAL代表的是串行,其实这里官方文档中解释NULL也代表串行:

/*!
 * @const DISPATCH_QUEUE_SERIAL
 * @discussion A dispatch queue that invokes blocks serially in FIFO order.
 */
#define DISPATCH_QUEUE_SERIAL NULL

b.任务创建并放到队列中

    for (int i = 0; i < 5; ++i) {
        dispatch_async(queue, ^{
            [self operation:i];
        });
    }
    NSLog(@"串行异步over again");

这里的dispatch_async表示的异步。
打印日志如下

串行异步over again
NSThread: 0x7fad03d53e30>{number = 2, name = (null)} ~ 0
<NSThread: 0x7fad03d53e30>{number = 2, name = (null)} ~ 1
<NSThread: 0x7fad03d53e30>{number = 2, name = (null)} ~ 2
<NSThread: 0x7fad03d53e30>{number = 2, name = (null)} ~ 3
<NSThread: 0x7fad03d53e30>{number = 2, name = (null)} ~ 4

从打印日志可以看出,串行异步,先执行的是串行异步over again,然后再去在子线程中一个一个的执行任务。

1.3并发同步

并发是多个任务可以同时执行的,这当然需要是在多线程中的。但是同步又要求的是在同一个线程中进行。
并发dispatch queue可以同时并行地执行多个任务,不过并发queue仍然按先进先出的顺序来启动任务。并发queue会在之前的任务完成之前就出列下一个任务并开始执行。并发queue同时执行的任务数量会根据应用和系统动态变化,各种因素包括:可用核数量、其它进程正在执行的工作数量、其它串行dispatch queue中优先任务的数量等。
a.创建队列

dispatch_queue_t queue = dispatch_queue_create("并发同步", DISPATCH_QUEUE_CONCURRENT);

b.将任务放到队列中

for (int i = 0; i < 5; ++i) {
        dispatch_sync(queue, ^{
            //
            [self operation:i];
        });
    }
    NSLog(@"并发同步over");

打印日志:

<NSThread: 0x7f9be8c05c20>{number = 1, name = main}
<NSThread: 0x7f9be8c05c20>{number = 1, name = main} ~ 0
<NSThread: 0x7f9be8c05c20>{number = 1, name = main} ~ 1
<NSThread: 0x7f9be8c05c20>{number = 1, name = main} ~ 2
<NSThread: 0x7f9be8c05c20>{number = 1, name = main} ~ 3
<NSThread: 0x7f9be8c05c20>{number = 1, name = main} ~ 4
并发同步over

可以看出这和串行同步的打印日志几乎一致,由于是串行,所以就只能在线程中一个一个执行任务,也就造成了和串行同步一致的效果。

1.4异步并发

异步并发可以是多个任务在不同的线程中同时执行。

     // 1.创建队列  
     dispatch_queue_t queue = dispatch_queue_create("异步并发", DISPATCH_QUEUE_CONCURRENT);
    // 2.将任务放到队列中异步执行
    for (int i = 0; i < 5; ++i) {
        dispatch_async(queue, ^{
            [self operation:i];
        });
    }
    NSLog(@"异步并发over again");

打印日志:

异步并发over again
<NSThread: 0x7ff5f1cca510>{number = 3, name = (null)} ~ 1
<NSThread: 0x7ff5f1d173b0>{number = 2, name = (null)} ~ 0
<NSThread: 0x7ff5f1d0e980>{number = 4, name = (null)} ~ 2
<NSThread: 0x7ff5f1cca510>{number = 3, name = (null)} ~ 3
<NSThread: 0x7ff5f1d173b0>{number = 2, name = (null)} ~ 4

可以看出首先打印的是异步并发over again,然后在回调执行的任务,而任务的执行也是在不同的线程中同时执行的。而迅雷下载也就是应用的这种方式。

2. 全局队列

全局队列:GCD默认创建的一个并发队列,使用的时候只需要获取

a.同步

 for (int i = 0; i < 20; ++i) {
        dispatch_sync(dispatch_get_global_queue(0, 0), ^{
            [self operation:i];
        });
    }

这里的 dispatch_get_global_queue(0, 0) 有两个参数:
参数1:队列的优先级,优先级的有:
DISPATCH_QUEUE_PRIORITY_HIGH:
DISPATCH_QUEUE_PRIORITY_DEFAULT:
DISPATCH_QUEUE_PRIORITY_LOW:
DISPATCH_QUEUE_PRIORITY_BACKGROUND:
参数2:预留参数,至今也未用。
打印日志:

<NSThread: 0x7fa1da408ab0>{number = 1, name = main} ~ 0
<NSThread: 0x7fa1da408ab0>{number = 1, name = main} ~ 1
<NSThread: 0x7fa1da408ab0>{number = 1, name = main} ~ 2
<NSThread: 0x7fa1da408ab0>{number = 1, name = main} ~ 3
<NSThread: 0x7fa1da408ab0>{number = 1, name = main} ~ 4

这结果又和并发同步执行的结果是一致的。

b.异步

   for (int i = 0; i < 20; ++i) {
        dispatch_async(dispatch_get_global_queue(0, 0), ^{
            [self operation:i];
        });
    }

执行结果

<NSThread: 0x7feac840ff10>{number = 2, name = (null)} ~ 0
<NSThread: 0x7feac840d2a0>{number = 3, name = (null)} ~ 1
<NSThread: 0x7feac840ff10>{number = 2, name = (null)} ~ 2
<NSThread: 0x7feac840d2a0>{number = 3, name = (null)} ~ 3
<NSThread: 0x7feac840ff10>{number = 2, name = (null)} ~ 4

看出执行的结果和异步并发的结果是一致的,是多个任务在多个线程中执行的。

3.主队列

主队列是直接和主线程关联的队列,如果将一个任务添加到主队列中,那么这个任务就只能在主线程中执行;主队列默认已经被创建好,使用的时候只需要获取。

a.同步

for (int i = 0; i < 20; ++i) {
 dispatch_sync(dispatch_get_main_queue(), ^{
   [self operation:i];
 });
}

分析:异步执行,需要在不同的线程中执行,就需要开不同的线程,而这又是主队列,所以只能在主线程中执行,所以会造成线程冲突或形成死锁。
而实际的运行结果也证实了。

b.异步

首先获取队列:

dispatch_queue_t queue = dispatch_get_main_queue();

将任务添加到主队列中异步执行

for (int i = 0; i < 20; ++i) {
    dispatch_async(queue, ^{
        [self operation:i];
    });
}

打印日志:

<NSThread: 0x7fa239402580>{number = 1, name = main} ~ 0
<NSThread: 0x7fa239402580>{number = 1, name = main} ~ 1
<NSThread: 0x7fa239402580>{number = 1, name = main} ~ 2
<NSThread: 0x7fa239402580>{number = 1, name = main} ~ 3
<NSThread: 0x7fa239402580>{number = 1, name = main} ~ 4

这是在主线程中一个一个按顺序执行。

2.4线程中的通信

设想下,我们进行网络请求的时候,假如有数据需要处理的时候,如果使用的是同步串行的这种方式,那这个体验度肯定是很糟糕的,而我们可以在子线程中进行下载数据,然后回到主线程中进行展示数据,而且这只能是异步执行的。这中情况就需要线程之间进行通信了。

    // 1.下载图片
    // 图片的下载和展示在全局队列中执行
    dispatch_async(dispatch_get_global_queue(0, 0), ^{
        NSLog(@"download:%@",[NSThread currentThread]);
       // 2.下载图片任务
        NSData *imageData = [NSData dataWithContentsOfURL:[NSURL URLWithString:@"http://attachment.bbs.ptbus.com/data/attachment/forum/201110/15/012639iwopzy3si06y5pbs.jpg"]];
        //
        UIImage *image = [UIImage imageWithData:imageData];
        
        // 在这儿图片下载结束,需要在这里回到主线程
        dispatch_async(dispatch_get_main_queue(), ^{
            NSLog(@"show:%@",[NSThread currentThread]);
            // 1.展示下载好的图片
            self.imageView.image = image;
        });
    });

打印的日志

download:<NSThread: 0x7fece24b4860>{number = 2, name = (null)}
show:<NSThread: 0x7fece2407e20>{number = 1, name = main}

可以清楚的看到下载任务是在子线程中进行的,而展示数据是在主线程进行的。
下载的图片也展示出来了。


线程间的通信.png

2.5队列组

上面只是执行单个任务,如果是很多的任务的时候,就需要用到队列组。队列组的作用就是:如果“希望多个异步执行任务,都执行完成后再回到主线程”的问题。
假如要求两个任务同时执行,如果用异步执行,下载完成回到主线程的方案是可以解决问题,但其实两个任务的过程并不需要按顺序执行,并发执行它们可以提高执行速度。有个注意点就是必须等两个任务都执行完毕后才能回到主线程显示的。而Dispatch Group能够在这种情况下帮我们提升性能。
我们可以使用dispatch_group_async函数将多个任务关联到一个Dispatch Group和相应的queue中,group会并发地同时执行这些任务。而且Dispatch Group可以用来阻塞一个线程, 直到group关联的所有的任务完成执行。有时候你必须等待任务完成的结果,然后才能继续后面的处理。
a.创建一个队列组

dispatch_group_t group = dispatch_group_create();

dispatch_group_t是一个队列组类型。

b.将任务间接放到队列组中,然后异步执行

    /**
     *  参数1:队列组
     *  参数2:队列
     *  参数2:任务
     */
    dispatch_group_async(group, dispatch_get_global_queue(0, 0), ^{
        NSLog(@"A:%@",[NSThread currentThread]);
        // 任务A
        NSLog(@"down load movie A");
    });
    dispatch_group_async(group, dispatch_get_global_queue(0, 0), ^{
        NSLog(@"B:%@",[NSThread currentThread]);
        // 任务B
        NSLog(@"down load movie B");
    });

c.两个任务结束后回到主线程

    dispatch_group_notify(group, dispatch_get_main_queue(), ^{
       // 回到主线程要执行的任务
        NSLog(@"%@",[NSThread currentThread]);
        NSLog(@"back to main thread");
    });

打印结果

A:<NSThread: 0x7fb93a51abc0>{number = 2, name = (null)}
B:<NSThread: 0x7fb93a5bcdd0>{number = 3, name = (null)}
down load movie A
down load movie B
<NSThread: 0x7fb93a507ef0>{number = 1, name = main}
back to main thread

可以知晓这两个任务分别是在不同的线程中执行的,最后都是回到主线程展示的。

总结

1.同步、异步、串行和并发的四种组合。

  1. 全局队列和主队列。
    3.队列组。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,590评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 86,808评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,151评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,779评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,773评论 5 367
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,656评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,022评论 3 398
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,678评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 41,038评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,659评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,756评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,411评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,005评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,973评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,203评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,053评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,495评论 2 343

推荐阅读更多精彩内容