GCD中判断线程内容执行完毕

前言:

在使用GCD多线程做操作时,有些时候需要咱们多线程里面的内容全部完成以后,再去刷新页面内容,这种情况下就需要咱们知道什么时候多线程队列里面的内容已经执行完毕,这里列举三种方法,希望能对有需要的小伙伴有启发。

一:信号量

信号量是一个整形值并且具有一个初始计数值,并且支持两个操作:信号通知和等待。当一个信号量被信号通知,其计数会被增加。当一个线程在一个信号量上等待时,线程会被阻塞(如果有必要的话),直至计数器大于零,然后线程会减少这个计数。

在GCD中有三个函数是semaphore的操作,分别是:
dispatch_semaphore_create   创建一个semaphore
dispatch_semaphore_signal   发送一个信号
dispatch_semaphore_wait    等待信号

第一个函数是创建一个信号量,我们会给它一个初始值,第二个函数是发送信号量,当调用该函数的时候,信号量的值会加一,第三方函数是等待信号量,调用等待信号量以后信号量的值会减一,并且当信号量的值小于0时,线程就会一直等待,直到信号量的值大于等于0时才能继续执行。

咱们下面逐一介绍这三个函数的调用以及参数意义

1.dispatch_semaphore_create

dispatch_samaphore_t dispatch_semaphore_create(long value);
  传入的参数为long,输出一个dispatch_semaphore_t类型且值为value的信号量。
 值得注意的是,这里的传入的参数value必须大于或等于0,否则dispatch_semaphore_create会返回NULL。

2.dispatch_semaphore_signal

    long dispatch_semaphore_signal(dispatch_semaphore_tdsema)
    这里的参数是第一个函数创建的信号
  1. dispatch_semaphore_wait

    long dispatch_semaphore_wait(dispatch_semaphore_t dsema,  dispatch_time_t timeout);第一参数的是信号,第二个参数是超时时间的设定,系统给我们了两         种超时时间设定,一种是 DISPATCH_TIME_NOW  表示当前; DISPATCH_TIME_FOREVER  表示遥远的未来;你可以选择其中一种也可以自己创建一个超时时间函数
    

创建超时timeout有两种方法

第一种方法:

dispatch_time_t dispatch_time(dispatch_time_t when, int64_t delta);
其参数when需传入一个dispatch_time_t类型的变量,和一个delta值。表示when加delta时间就是timeout的时间。

第二种方式:

dispatch_time_t dispatch_walltime(<#const struct timespec * _Nullable when, int64_t delta);
 其参数when需传入一个dispatch_time_t类型的变量,和一个delta值。表示when加delta时间就是timeout的时间。

6.关于信号量,一般可以用停车来比喻。

网吧剩余6个座位,那么即使同时来了6个人也能同时接待。如果此时来了7个人,那么就有一个人需要等待。

信号量的值就相当于剩余座位的数目,dispatch_semaphore_wait函数就相当于来了一个人,dispatch_semaphore_signal

就相当于走了一个人。网络座位的剩余数目在初始化的时候就已经指明了(dispatch_semaphore_create(long value)),

调用一次dispatch_semaphore_signal,剩余的座位就增加一个;调用一次dispatch_semaphore_wait剩余座位就减少一个;

当剩余座位为0时,再来人(即调用dispatch_semaphore_wait)就只能等待。有可能同时有几个人等待一个座位。有些人

没有耐心,给自己设定了一段等待时间,这段时间内等不到座位就走了,如果等到了就去上网。而有些人就想站在这里,所以就一直等下去。

具体到代码中
假设咱们现在需要做完A、B、事件以后才能去做C事件那么咱们就可以仿照下面的做法

#import "ViewController.h"
#import <objc/runtime.h>



@interface ViewController ()

@end

@implementation ViewController

- (void)viewDidLoad {
[super viewDidLoad];
[self initlizeAppeareces];
}


 - (void)initlizeAppeareces{
//crate的value表示,最多几个资源可访问
dispatch_semaphore_t semaphore = dispatch_semaphore_create(2);
dispatch_queue_t quene = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);

//任务1
dispatch_async(quene, ^{
    dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER);
    NSLog(@"办理A事件");
    dispatch_semaphore_signal(semaphore);
});
//任务2
dispatch_async(quene, ^{
    dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER);
    NSLog(@"办理B事件");
    dispatch_semaphore_signal(semaphore);
});

}

当咱们信号量为2时,证明之前的事件都办理完了,咱们就可以做其他的事情了。


二:队列加group分组
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(), ^{

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

});

当程序执行dispatch_group_notify()函数时,证明队列中的任务已经执行完了。

第三种方式dispatch_barrier_sync和dispatch_barrier_async

共同点:

1、等待在它前面插入队列的任务先执行完

2、等待他们自己的任务执行完再执行后面的任务

不同点:

1、dispatch_barrier_sync将自己的任务插入到队列的时候,需要等待自己的任务结束之后才会继续插入被写在它后面的任务,然后执行它们

2、dispatch_barrier_async将自己的任务插入到队列之后,不会等待自己的任务结束,它会继续把后面的任务插入到队列,然后等待自己的任务结束后才执行后面任务。

所以调用这两种方法中的其中一个都会等前面的方法执行完以后再执行之后的方法,就可以确定队列中任务是否完成

注: 在信号量的demo编写过程中我发现当信号量等于0的时候都崩溃了,如果有知道原因的老铁欢迎前来分享。

四:将任务添加到串行队列中

除了像上面那样处理,也可以把任务放到各个串行队列中,并用dispatch group跟踪其执行状况。然而,如果所有任务都排在同一个串行队列里面,那么dispatch group就用处不大了。因为此时任务总要逐个执行,所以只需要在提交完全部任务之后再提交一个块即可,这样做与通过notify函数等待dispatch group执行完毕然后再回调块是等效的:

dispatch_queue_t queue = dispatch_queue_create("com.effectiveobjectivec.queue",NULL);
for(id object in collection){
   dispatch_async(queue,^{
    [object performTask];
   });
  }
 dispatch_async(queue,^{
    //Continue processing after completing tasks
});

上面这段代码表明,开发者未必总需要使用dispatch group。有时候采用单个队列搭配标准的异步派发,也可以实现同样的效果。
笔者为何要在标题中谈到"根据系统资源状况来执行任务"呢?回头看看向并发队列派发任务的那个例子,就会明白了。为了执行队列中的块,GCD会在适当的实际自动创建新线程。

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

推荐阅读更多精彩内容