GCD 中的线程组 Group

配图

前言

本文主要介绍 GCD 中的线程组 Group,不讲 GCD 基础概念知识。如果对 GCD 的基本知识点不是很清楚的话,建议去补充一下。好了,废话不多说,坐稳了,马上就开车了。

正文

线程组

GCD 为我们提供了 dispatch_group 方法,它有一个组的概念,可以把多个任务归并到一个组内来执行,通过监听组内所有任务的执行情况来做相应处理。

1.线程组内的任务是同步的。

假设我们现在有两个任务,任务 1 和任务 2,任务 1:for 循环 1000 次,任务 2:for 循环 100 次。等到任务 1 和任务 2 都执行完后再执行回调。

上代码。

// 创建一个group
dispatch_group_t group = dispatch_group_create();
// 创建一个队列:全局队列
dispatch_queue_t queue = dispatch_get_global_queue(0, 0);

// 将任务1添加到 group 中    
dispatch_group_async(group, queue, ^{
   for (int i = 0; i < 1000; i++) {
       NSLog(@"任务1-----%d",i);
   }
});
 
// 将任务2添加到 group 中    
dispatch_group_async(group, queue, ^{
   for (int i = 0; i < 100; i++) {
       NSLog(@"任务2-----%d",i);
   }
});
    
// 任务1和任务2执行结束,回调
dispatch_group_notify(group, queue, ^{
  dispatch_async(dispatch_get_main_queue(), ^{
      NSLog(@"完成任务");
  });
});

打印结果

2016-12-16 12:31:34.723 Group[12091:514693] 任务1-----0
2016-12-16 12:31:34.723 Group[12091:514681] 任务2-----0
2016-12-16 12:31:34.723 Group[12091:514693] 任务1-----1
2016-12-16 12:31:34.723 Group[12091:514681] 任务2-----1
2016-12-16 12:31:34.723 Group[12091:514693] 任务1-----2
2016-12-16 12:31:34.723 Group[12091:514681] 任务2-----2
····
····
····
2016-12-16 12:31:35.105 Group[12091:514693] 任务1-----997
2016-12-16 12:31:35.105 Group[12091:514693] 任务1-----998
2016-12-16 12:31:35.105 Group[12091:514693] 任务1-----999
2016-12-16 12:31:35.106 Group[12091:514347] 完成任务

从打印结果可以看出:

先并发执行任务 1 和任务 2,任务 2 首先完成,然后任务 1 还在执行,任务 1 执行结束后,再执行回调。

2.线程组内的任务是异步的。

实际开发中我们可能有这样的需求:并发请求多个网络接口,等到所有的接口请求结束之后,我们再来个回调刷新 TableView。
这里面有些同学可能会说把前面例子里的任务 1 和 任务 2 改为网络请求就可以了。那好,那我们来试试。

上代码。

// 创建一个group
dispatch_group_t group = dispatch_group_create();
// 创建一个队列:全局队列
dispatch_queue_t queue = dispatch_get_global_queue(0, 0);

// 将任务1添加到 group 中    
dispatch_group_async(group, queue, ^{
   // 模拟异步网络请求
   dispatch_async(queue, ^{
       for (int i = 0; i < 1000; i++) {
           NSLog(@"任务1-----%d",i);
       }
   });
});
 
// 将任务2添加到 group 中    
dispatch_group_async(group, queue, ^{
   // 模拟异步网络请求
   dispatch_async(queue, ^{
       for (int i = 0; i < 100; i++) {
           NSLog(@"任务2-----%d",i);
       }
   });
});
    
// 任务1和任务2执行结束,回调
dispatch_group_notify(group, queue, ^{
  dispatch_async(dispatch_get_main_queue(), ^{
      NSLog(@"完成任务");
  });
});

打印结果

2016-12-16 13:12:40.541 Group[12636:545117] 任务1-----0
2016-12-16 13:12:40.541 Group[12636:544738] 完成任务
2016-12-16 13:12:40.541 Group[12636:545118] 任务2-----0
2016-12-16 13:12:40.541 Group[12636:545117] 任务1-----1
2016-12-16 13:12:40.541 Group[12636:545118] 任务2-----1
2016-12-16 13:12:40.542 Group[12636:545117] 任务1-----2
2016-12-16 13:12:40.542 Group[12636:545118] 任务2-----2
····
····
····
2016-12-16 13:12:40.876 Group[12636:545117] 任务1-----997
2016-12-16 13:12:40.876 Group[12636:545117] 任务1-----998
2016-12-16 13:12:40.877 Group[12636:545117] 任务1-----999

从打印结果可以看出:

回调并没有等到任务 1 和任务 2 执行完就打印了,怎么跟我们想得不一样呢?

好,那我下面来解释一下。

  1. 第一个例子中,任务 1 是同步的任务,任务 2 也是同步的任务。
  • 第二个例子中,任务 1 是异步的任务,任务 2 也是异步的任务。

同步和异步的最大区别是:同步是一个一个的执行,会有一个等待。而异步则不是,它不会等待。

因为 dispatch_group_async 里面的任务是异步的,所以任务在执行的时候,它不会去等待 for 循环执行结束,它会直接跳过 dispatch_async 这 block 执行下一句去了,所以 dispatch_group_notify 也会很快就执行。

下面再看下如何去解决这个问题吧。

这边就用到了 dispatch_group_enter 和 dispatch_group_leave。它们两个是成对出现的。dispatch_group_enter 使 group 里正要执行的任务数递增,dispatch_group_leave 则使之递减。所以调用完 dispatch_group_enter 以后,必须有与之对应的 dispatch_group_leave 才行。如果调用 dispatch_group_enter 之后,没有相应的 dispatch_group_leave 操作,那么这一组任务就永远执行不完。在使用时,可以在向队列中添加任务时调用 dispatch_group_enter,在任务执行完成之后合适的地方调用 dispatch_group_leave。

上代码

// 创建一个group
dispatch_group_t group = dispatch_group_create();
// 创建一个队列:全局队列
dispatch_queue_t queue = dispatch_get_global_queue(0, 0); 

// 进入group    
dispatch_group_enter(group);

// 模拟异步网络请求
dispatch_async(queue, ^{
   
   for (int i = 0; i < 1000; i++) {
       NSLog(@"任务1-----%d",i);
   }
   // 离开group
   dispatch_group_leave(group);
   
});
    
// 进入group     
dispatch_group_enter(group);
    
// 模拟异步网络请求
dispatch_async(queue, ^{
   
   for (int i = 0; i < 100; i++) {
       NSLog(@"任务2-----%d",i);
   }
   // 离开group
   dispatch_group_leave(group);
   
});
    
    
dispatch_group_notify(group, queue, ^{
   dispatch_async(dispatch_get_main_queue(), ^{
       NSLog(@"完成任务");
   });
});

打印结果

2016-12-16 13:15:17.945 Group[12870:559844] 任务1-----0
2016-12-16 13:15:17.945 Group[12870:559831] 任务2-----0
2016-12-16 13:15:17.945 Group[12870:559844] 任务1-----1
2016-12-16 13:15:17.945 Group[12870:559831] 任务2-----1
2016-12-16 13:15:17.945 Group[12870:559844] 任务1-----2
2016-12-16 13:15:17.946 Group[12870:559831] 任务2-----2
····
····
····
2016-12-16 13:15:18.375 Group[12870:559844] 任务1-----997
2016-12-16 13:15:18.376 Group[12870:559844] 任务1-----998
2016-12-16 13:15:18.376 Group[12870:559844] 任务1-----999
2016-12-16 13:15:18.376 Group[12870:559491] 完成任务

从打印结果可以看出:

和例子一的打印结果一样,OK,问题解决。

其实还有一个方法也可以解决,信号量 dispatch_semaphore。

什么是信号量?

引用网友举的一个例子:

以一个停车场的运作为例。简单起见,假设停车场只有三个车位,一开始三个车位都是空的。这时如果同时来了五辆车,看门人允许其中三辆直接进入,然后放下车拦,剩下的车则必须在入口等待,此后来的车也都不得不在入口处等待。这时,有一辆车离开停车场,看门人得知后,打开车拦,放入外面的一辆进去,如果又离开两辆,则又可以放入两辆,如此往复。在这个停车场系统中,车位是公共资源,每辆车好比一个线程,看门人起的就是信号量的作用。

抽象的来讲,信号量的特性如下:信号量是一个非负整数(车位数),所有通过它的线程/进程(车辆)都会将该整数减一(通过它当然是为了使用资源),当该整数值为零时,所有试图通过它的线程都将处于等待状态。在信号量上我们定义两种操作: Wait(等待) 和 Release(释放)。当一个线程调用Wait操作时,它要么得到资源然后将信号量减一,要么一直等下去(指放入阻塞队列),直到信号量大于等于一时。Release(释放)实际上是在信号量上执行加操作,对应于车辆离开停车场,该操作之所以叫做“释放”是因为释放了由信号量守护的资源。

在 GCD 中有三个函数是 semaphore 的操作,分别是:

// 创建一个semaphore
dispatch_semaphore_create(long value);   

// 发送一个信号
dispatch_semaphore_signal(dispatch_semaphore_t dsema);  

// 等待信号 
dispatch_semaphore_wait(dispatch_semaphore_t dsema, dispatch_time_t timeout);  

上代码

// 创建一个group
dispatch_group_t group = dispatch_group_create();
// 创建一个队列:全局队列
dispatch_queue_t queue = dispatch_get_global_queue(0, 0);

// 创建信号量,并且设置值为0    
dispatch_semaphore_t semaphore = dispatch_semaphore_create(0);

// 将任务1添加到 group 中    
dispatch_group_async(group, queue, ^{
   
   // 模拟异步网络请求
   dispatch_async(queue, ^{
       for (int i = 0; i < 1000; i++) {
           NSLog(@"任务1-----%d",i);
       }

        // 每次发送信号则 semaphore 会 +1
       dispatch_semaphore_signal(semaphore);

   });
   // 由于是异步执行的,当 semaphore 等于 0,则会阻塞当前线程,直到执行了 block 的 dispatch_semaphore_signal,semaphore + 1,才会继续执行。
   dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER);
   
});
 
// 将任务2添加到 group 中        
dispatch_group_async(group, queue, ^{
   
   // 模拟异步网络请求
   dispatch_async(queue, ^{
       for (int i = 0; i < 100; i++) {
           NSLog(@"任务2-----%d",i);
       }
       
       dispatch_semaphore_signal(semaphore);

   });

   dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER);
});
    
    
dispatch_group_notify(group, queue, ^{
   dispatch_async(dispatch_get_main_queue(), ^{
       NSLog(@"完成任务");
   });
});

由于是异步执行的,当 semaphore 等于 0,则会阻塞当前线程,直到执行了 block 的 dispatch_semaphore_signal,semaphore + 1,才会继续执行。这样很好的解决了这个问题。

2016-12-16 13:18:11.747 Group[13276:586281] 任务1-----0
2016-12-16 13:18:11.747 Group[13276:586296] 任务2-----0
2016-12-16 13:18:11.747 Group[13276:586281] 任务1-----1
2016-12-16 13:18:11.747 Group[13276:586296] 任务2-----1
2016-12-16 13:18:11.747 Group[13276:586281] 任务1-----2
2016-12-16 13:18:11.748 Group[13276:586296] 任务2-----2
····
····
····
2016-12-16 13:18:12.111 Group[13276:586281] 任务1-----997
2016-12-16 13:18:12.111 Group[13276:586281] 任务1-----998
2016-12-16 13:18:12.111 Group[13276:586281] 任务1-----999
2016-12-16 13:18:12.111 Group[13276:585910] 完成任务

从打印结果可以看出:

和例子一的打印结果还是一样。

上面两个方法都可以,可能还有更多的方法。笔者暂时只知道这两个方法,如果有大神知道还有其他的方法,可以联系我,一起交流。

最后

由于笔者水平有限,文中如果有错误的地方,还望大神指出。或者有更好的方法和建议,我们可以一起交流。
附上本文的所有 demo 下载链接【GitHub】,配合 demo 一起看文章,效果会更佳。

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

推荐阅读更多精彩内容