GCD(队列组、信号量、栅栏函数)实现多个请求都完成之后返回结果

队列组

队列组的简单使用 -- 监听任务的完成

1、所有的任务会并发的执行(不按序)

2、所有的异步函数, 都是添加到队列中, 然后再纳入到队列组的监听范围

3、使用dispatch_group_notify(队列组, 队列)函数, 来监听在这个函数上面的任务执行是否完成, 当任务完成, 就会调用这个方法

// 1. 创建队列组
dispatch_group_t group = dispatch_group_create(); 
// 2. 创建并发队列 
dispatch_queue_t queue = dispatch_queue_create("123", DISPATCH_QUEUE_CONCURRENT); 
// 3. 使用函数添加任务 
dispatch_group_async(group, queue, ^{ 
    NSLog(@"1---%@", [NSThread currentThread]);
 }); 
dispatch_group_async(group, queue, ^{
     NSLog(@"2---%@", [NSThread currentThread]); 
});
 dispatch_group_async(group, queue, ^{ 
    NSLog(@"3---%@", [NSThread currentThread]); 
}); 
dispatch_group_async(group, queue, ^{ 
    NSLog(@"4---%@", [NSThread currentThread]); 
}); 
// 4. 让队列组监听任务的完成 
dispatch_group_notify(group, queue, ^{ 
    NSLog(@"执行完毕"); 
});

信号量

问题描述:

假设现在系统有两个空闲资源可以被利用,但同一时间却有三个线程要进行访问,这种情况下,该如何处理呢?

或者

我们要下载很多图片,并发异步进行,每个下载都会开辟一个新线程,可是我们又担心太多线程cpu肯定吃不消,那么我们这里也可以用信号量控制一下最大开辟线程数。

定义:

1、信号量:就是一种可用来控制访问资源的线程数量的标识,设定了一个信号量,在线程访问之前,加上信号量的处理,则可告知系统按照我们指定的信号量数量来执行多个线程。

其实,这有点类似锁机制了,只不过信号量都是系统帮助我们处理了,我们只需要在执行线程之前,设定一个信号量值,并且在使用时,加上信号量处理方法就行了。

2、信号量主要有3个函数,分别是:

//创建信号量,参数:信号量的初值,如果小于0则会返回NULL
dispatch_semaphore_create(信号量值)
//等待降低信号量
dispatch_semaphore_wait(信号量,等待时间)
//提高信号量
dispatch_semaphore_signal(信号量)

注意,正常的使用顺序是先降低然后再提高,这两个函数通常成对使用。 (具体可参考下面的代码示例)

3、那么就开头提的问题,我们用代码来解决

-(void)dispatchSignal{

    //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(@"run task 1");

        sleep(1);

        NSLog(@"complete task 1");

        dispatch_semaphore_signal(semaphore);        

    });

    //任务2

    dispatch_async(quene, ^{

        dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER);

        NSLog(@"run task 2");

        sleep(1);

        NSLog(@"complete task 2");

        dispatch_semaphore_signal(semaphore);        

    });

    //任务3

    dispatch_async(quene, ^{

        dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER);

        NSLog(@"run task 3");

        sleep(1);

        NSLog(@"complete task 3");

        dispatch_semaphore_signal(semaphore);        

    });    

}

执行结果:


image.png

总结:由于设定的信号值为2,先执行两个线程,等执行完一个,才会继续执行下一个,保证同一时间执行的线程数不超过2。

这里我们扩展一下,假设我们设定信号值=1

dispatch_semaphore_create(1)

那么结果就是:

image.png

如果设定信号值=3

dispatch_semaphore_create(3)

那么结果就是:

image.png

其实设定为3,就是不限制线程执行了,因为一共才只有3个线程。

栅栏函数

当我们的任务有依赖关系的时候,比如任务1和2执行完毕后才能执行任务3和4,这时候我们可以用到这个函数——栅栏函数。其中 queue 是队列,block 是任务。

提交一个栅栏函数在同步执行中,它会等待栅栏函数执行完再去执行下一行代码(注意是下一行代码),同步栅栏函数是在当前线程中执行的

dispatch_barrier_sync(dispatch_queue_t queue, dispatch_block_t blcok);

提交一个栅栏函数在异步执行中,它会立马返回开始执行下一行代码(不用等待任务执行完毕)

dispatch_barrier_async(dispatch_queue_t queue, dispatch_block_t blcok);
共同点

都会等待在它前面插入队列的任务(1、2、3)先执行完 2、都会等待他们自己的任务(barrier)执行完再执行后面的任务(4、5、6)(注意这里说的是任务不是下一行代码)

不同点

dispatch_barrier_sync需要等待自己的任务(barrier)结束之后,才会继续添加并执行写在barrier后面的任务(4、5、6),然后执行后面的任务 2、dispatch_barrier_async将自己的任务(barrier)插入到queue之后,不会等待自己的任务结束,它会继续把后面的任务(4、5、6)插入到queue,然后执行任务。

// 并发队列 栅栏函数 
- (void)concurrentQueueAsyncAndSync2BarrierTest {
    dispatch_queue_t queue = dispatch_queue_create("com.barrier", DISPATCH_QUEUE_CONCURRENT);
    
    dispatch_async(queue, ^{
        NSLog(@"任务0 start");
        sleep(1);
        NSLog(@"任务0 end");
    });
    
    dispatch_async(queue, ^{
        NSLog(@"任务1 start");
        sleep(1);
        NSLog(@"任务1 end");
    });
    
    NSLog(@"同步栅栏 start 😄");
    dispatch_barrier_sync(queue, ^{
        NSLog(@"同步栅栏 任务 start");
        sleep(1);
        NSLog(@"同步栅栏 任务 end");
    });
    NSLog(@"同步栅栏 end 😄");
    
    dispatch_async(queue, ^{
        NSLog(@"任务2 start");
        sleep(1);
        NSLog(@"任务2 end");
    });
    
    NSLog(@"异步栅栏 start 😄");
    dispatch_barrier_async(queue, ^{
        NSLog(@"异步栅栏栅栏 任务 start");
        sleep(1);
        NSLog(@"异步栅栏栅栏 任务 end");
    });
    NSLog(@"异步栅栏栅栏 end 😄");
    
    dispatch_async(queue, ^{
        NSLog(@"任务3 start");
        sleep(1);
        NSLog(@"任务3 end");
    });
    
    dispatch_async(queue, ^{
        NSLog(@"任务4 start");
        sleep(1);
        NSLog(@"任务4 end");
    });
}

打印结果如下:

2019-03-17 16:17:50.447824+0800 网络请求Demo[3358:203368] 同步栅栏 start 😄
2019-03-17 16:17:50.447838+0800 网络请求Demo[3358:203589] 任务0 start
2019-03-17 16:17:50.447871+0800 网络请求Demo[3358:203806] 任务1 start
2019-03-17 16:17:51.451935+0800 网络请求Demo[3358:203806] 任务1 end
2019-03-17 16:17:51.451935+0800 网络请求Demo[3358:203589] 任务0 end
2019-03-17 16:17:51.452211+0800 网络请求Demo[3358:203368] 同步栅栏 任务 start
2019-03-17 16:17:52.452917+0800 网络请求Demo[3358:203368] 同步栅栏 任务 end
2019-03-17 16:17:52.453108+0800 网络请求Demo[3358:203368] 同步栅栏 end 😄
2019-03-17 16:17:52.453240+0800 网络请求Demo[3358:203368] 异步栅栏 start 😄
2019-03-17 16:17:52.453280+0800 网络请求Demo[3358:203809] 任务2 start
2019-03-17 16:17:52.453368+0800 网络请求Demo[3358:203368] 异步栅栏栅栏 end 😄
2019-03-17 16:17:53.067835+0800 网络请求Demo[3358:203368] GCDBarrierController
2019-03-17 16:17:53.458678+0800 网络请求Demo[3358:203809] 任务2 end
2019-03-17 16:17:53.458902+0800 网络请求Demo[3358:203809] 异步栅栏栅栏 任务 start
2019-03-17 16:17:54.462291+0800 网络请求Demo[3358:203809] 异步栅栏栅栏 任务 end
2019-03-17 16:17:54.462529+0800 网络请求Demo[3358:203809] 任务3 start
2019-03-17 16:17:54.462534+0800 网络请求Demo[3358:203806] 任务4 start
2019-03-17 16:17:55.465377+0800 网络请求Demo[3358:203806] 任务4 end
2019-03-17 16:17:55.465391+0800 网络请求Demo[3358:203809] 任务3 end
情景分析:

同步栅栏添加进入队列的时候,当前线程会被锁死,直到同步栅栏之前的任务和同步栅栏任务本身执行完毕时,当前线程才会打开然后继续执行下一句代码。

注意:

在使用栅栏函数时.使用自定义队列才有意义,如果用的是串行队列或者系统提供的全局并发队列,这个栅栏函数的作用等同于一个同步函数的作用

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

推荐阅读更多精彩内容