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

2019-03-17  本文已影响0人  nucky_lee

队列组

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

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
情景分析:

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

注意:

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

上一篇下一篇

猜你喜欢

热点阅读