iOS GCD 线程同步方法

我们使用GCD的时候如何让线程同步,目前我能想到的就三种

  • 1.dispatch_group
  • 2.dispatch_barrier
  • 3.dispatch_semaphore

一、dispatch_group 线程组

1.线程组,是一种同步机制,可以让某些线程先执行,某些线程最后执行,以控制线程的执行顺序。
2.有这么一个需要,分别执行2个耗时的异步操作,等2个异步操作都执行完毕后在回到主线程执行操作。如果想要快速高效地实现这个需求,可以考虑使用线程组。 线程组的创建代码如下:
(1.) 创建dispatch_group_t

 dispatch_group_t group = dispatch_group_create();

(2.) 往线程组里面添加任务的函数如下
自己创建队列:使用dispatch_group_async

无法直接使用队列变量(如使用AFNetworking添加异步任务):使用dispatch_group_enterdispatch_group_leave
调用了dispatch_group_enter(dispatch_group_t group)
之后,必须有与之对应的dispatch_group_leave(dispatch_group_t group)
才行
第一种:

  group 负责监控任务,queue 负责调度任务
 dispatch_group_async(group, dispatch_get_global_queue(0, 0), ^{  
     i = 1;  
 });  
  
 dispatch_group_async(group, dispatch_get_global_queue(0, 0), ^{  
     j = 2;  
 });  

第二种:

AFHTTPRequestOperationManager *manager = [AFHTTPRequestOperationManager manager];

//Enter group
dispatch_group_enter(group);
[manager GET:@"http://www.baidu.com" parameters:nil success:^(AFHTTPRequestOperation *operation, id responseObject) {
    //Deal with result...

    //Leave group
    dispatch_group_leave(group);
}    failure:^(AFHTTPRequestOperation *operation, NSError *error) {
    //Deal with error...

    //Leave group
    dispatch_group_leave(group);
}];

//More request...

(3.) 监听所有任务完成 - 等到 group 中的所有任务执行完毕后,"由队列调度 block 中的任务异步执行!"
在当前线程阻塞的同步等待:dispatch_group_wait。
添加一个异步执行的任务作为结束任务:dispatch_group_notify

 dispatch_group_notify(group, dispatch_get_global_queue(0, 0), ^{  
     NSLog(@"%d",i+j);  
 }); 

::适用于后台批量下载 结束后主线程统一刷新UI

二、dispatch_barrier 栅栏块

dispatch_barrier_async,对于同一个队列中的不同任务而言,在dispatch_barrier_async之前的先执行,在dispatch_barrier_async后面的后执行 .

_syncQueue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);

- (NSString *)someString
{
 __weak NSString *localSomeString;

 dispatch_sync(_syncQueue, ^{

 localSomeString = _someString;

 });

 return localSomeString;

}
- (void)setSomeString:(NSString *)someString
{
 // barrier
 dispatch_barrier_async(_syncQueue, ^{

 _someString = someString;

 });
}

函数 dispatch_barrier_sync 和 dispatch_barrier_async
可以让队列中派发的 block 变成 barrier(栅栏) 使用,这种 block 称为 barrier block。队列中的 barrier block 必须等当前并发队列中的 block 都执行结束才开始执行,时序图如下:

image_note64272_1.png

在这个并发队列中,读取操作是用普通的块来实现的,而写入操作则是用栅栏块来实现的,读取操作可以并行,但写入操作必须单独执行,因为他是栅栏快.
注意dispatch_barrier_async的同步控制和线程组、信号量的同步机制是不一样的,dispatch_barrier_async是对于同一个队列中的不同任务而言的,线程组和信号量是对于不同线程而言的。

三、dispatch_semaphore 信号量

创建一个信号量。参数指定信号量的起始值。这个数字是你可以访问的信号量,不需要先去增加它的数量(增加信号量也叫作发射信号量)。
初始value = 0时,信号量--,小于0,wait线程阻塞。然后执行signal,信号量++,激活wait线程。

    dispatch_semaphore_t sema = dispatch_semaphore_create(0);
    
    dispatch_async(dispatch_get_global_queue(0, 0), ^{
        
        dispatch_semaphore_wait(sema, DISPATCH_TIME_FOREVER);
    
        NSLog(@"等你");
    });
    dispatch_async(dispatch_get_global_queue(0, 0), ^{
       
        NSLog(@"发送信号");
        [NSThread sleepForTimeInterval:5];
        dispatch_semaphore_signal(sema);
        
    });
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • iOS 多线程系列 -- 基础概述iOS 多线程系列 -- pthreadiOS 多线程系列 -- NSThrea...
    shannoon阅读 4,365评论 0 2
  • 背景 担心了两周的我终于轮到去医院做胃镜检查了!去的时候我都想好了最坏的可能(胃癌),之前在网上查的症状都很相似。...
    Dely阅读 13,028评论 21 42
  • 目录:iOS多线程(一)--pthread、NSThreadiOS多线程(二)--GCD详解iOS多线程(三)--...
    Claire_wu阅读 4,707评论 0 6
  • 定义工厂方法模式定义了一个创建对象的接口,但有子类决定要实例化的类是哪一个。工厂方法让类把实例化推迟到子类。类图工...
    一一小知阅读 1,443评论 0 0
  • 夏意盎然,午后的教室,闷热正试图扼死空气中的最后一缕凉意,没有几个人甘愿在教室里坐以待毙,懒得出去的她耷拉在桌子上...
    哑刃阅读 2,118评论 1 3