iOS GCD基本使用及详解

GCD的基本函数:

  • dispatch_sync() 同步执行
  • dispatch_async() 异步执行
  • dispatch_after() 延时执行
  • dispatch_once() 一次性执行
  • dispatch_apply() 提交队列
  • dispatch_queue_create() 创建队列
  • dispatch_group_create() 创建队列组
  • dispatch_group_async() 提交任务到队列组
  • dispatch_group_enter() / dispatch_group_leave() 将队列组中的任务未执行完毕的任务数目加减1(两个函数要配合使用)
  • dispatch_group_notify() 监听队列组执行完毕
  • dispatch_group_wait() 设置等待时间(返回 0成功,1失败)

队列:

GCD 用dispatch queue来处理代码块,这些队列管理你提供给 GCD 的任务并执行这些任务。这就保证了第一个被添加到队列里的任务会是队列中第一个开始的任务,而第二个被添加的任务将第二个开始,如此直到队列的终点。

  • 主队列:系统提供的一个特殊队列。和其它串行队列一样,这个队列中的任务一次只能执行一个。它能保证所有的任务都在主线程执行,而主线程是唯一可用于更新 UI 的线程。这个队列就是用来在主线程上进行操作的。
  • 串行队列:串行队列中的任务一次执行一个,每个任务只会在上一个任务完成时才开始执行。
  • 并发队列:在并发队列中的任务会按照被添加的顺序开始执行,任务可能以任意顺序完成,你不能知道什么时开始执行下一个任务,或者有多少任务在同时执行。
//获取主队列
dispatch_queue_t mainQueue = dispatch_get_main_queue();
//使用dispatch_get_global_queue()获取全局并发队列,第一个参数是队列优先级,第二个参数传0.
dispatch_queue_t otherQueue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);
/** 
  * 使用 dispatch_queue_create 初始化 concurrentQueue 为一个并发队列。
  * 第一个参数是队列标识;第二个参数指定你的队列是串行还是并发。设为NULL时默认是DISPATCH_QUEUE_SERIAL,将创建串行队列.
  * 在必要情况下,你可以将其设置为DISPATCH_QUEUE_CONCURRENT来创建自定义并行队列.
 */
dispatch_queue_t myQueue = dispatch_queue_create("myQueue", DISPATCH_QUEUE_CONCURRENT);

同步和异步:

//同步函数,在当前线程执行(不开启新的线程)
dispatch_sync(otherQueue, ^{
    NSLog(@"同步:%@",[NSThread currentThread]);
});

//异步函数,开启子线程执行
dispatch_async(otherQueue, ^{
    NSLog(@"异步:%@",[NSThread currentThread]);
});
打印:
[15276:540462] 同步:<NSThread: 0x600000068680>{number = 1, name = main}
[15276:540519] 异步:<NSThread: 0x600000074a80>{number = 3, name = (null)}

串行队列:

一个任务执行完毕后,再执行下一个任务

  • 异步
// 1.使用 dispatch_queue_creat()创建串行队列
dispatch_queue_t serialQueue = dispatch_queue_create("serialQueue", NULL);

//开启新线程,串行执行任务
NSLog(@"异步函数执行串行队列,当前线程:%@",[NSThread currentThread]);
dispatch_async(serialQueue, ^{
    NSLog(@"任务1:%@",[NSThread currentThread]);
});

dispatch_async(serialQueue, ^{
    NSLog(@"任务2:%@",[NSThread currentThread]);
});

dispatch_async(serialQueue, ^{
    NSLog(@"任务3:%@",[NSThread currentThread]);
});
  • 同步
//不开启新线程,串行执行任务
NSLog(@"同步函数执行串行队列,当前线程:%@",[NSThread currentThread]);

dispatch_sync(serialQueue, ^{
    NSLog(@"任务1:%@",[NSThread currentThread]);
});

dispatch_sync(serialQueue, ^{
    NSLog(@"任务2:%@",[NSThread currentThread]);
});

dispatch_sync(serialQueue, ^{
    NSLog(@"任务3:%@",[NSThread currentThread]);
});
并发队列:
  • 多个任务并发执行(自动开启多个线程同时执行任务)
  • 并发功能只有在异步(dispatch_async)函数下才有效!
dispatch_queue_t concurrentQueue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);

//异步函数,并发队列
//开启新线程,并发执行任务
NSLog(@"异步函数执行并发队列,当前线程:%@",[NSThread currentThread]);
dispatch_async(concurrentQueue, ^{
    NSLog(@"任务1:%@",[NSThread currentThread]);
});

dispatch_async(concurrentQueue, ^{
    NSLog(@"任务2:%@",[NSThread currentThread]);
});

dispatch_async(concurrentQueue, ^{
   NSLog(@"任务3:%@",[NSThread currentThread]);
});
队列组:

任务1,任务2同时执行,所有任务都执行成功后回到主线程,高效率

NSLog(@"队列组执行任务,当前线程:%@",[NSThread currentThread]);
//1.创建队列组 dispatch_group_create()
dispatch_group_t group = dispatch_group_create();

//2.开启任务
//开启任务1
//提交任务到队列组 dispatch_group_async()
dispatch_group_async(group, dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
    for (int i = 0; i < 5; i++) {
        NSLog(@"任务1 :%@",[NSThread currentThread]);
    }
});

//开启任务2
dispatch_group_async(group, dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
    for (int i = 0; i < 5; i++) {
        NSLog(@"任务2 :%@",[NSThread currentThread]);
    }
});

//所有任务执行完毕,回到主线程进行操作
dispatch_group_notify(group, dispatch_get_main_queue(), ^{
    NSLog(@"任务1,2执行完毕,回到主线程:%@",[NSThread currentThread]);
});

延时执行:

延迟一段时间把一项任务提交到队列中执行,返回之后就不能取消,常用来在主队列上延迟执行一项任务。

NSLog(@"当前线程 %@", [NSThread currentThread]);
//GCD延时调用(主线程)(主队列)
/**
 * 1.声明一个变量 afterTime 指定要延迟的时长
 * 2.等待 afterTime 给定的时长,再异步地添加一个 Block 到主线程。
 */
dispatch_time_t afterTime = dispatch_time(DISPATCH_TIME_NOW, (int64_t)(2 * NSEC_PER_SEC));//1

dispatch_after(afterTime, dispatch_get_main_queue(), ^{//2
    NSLog(@"GCD延时调用(主线程):%@",[NSThread currentThread]);
});

//GCD延时调用(其他线程)(全局并发队列)
dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(3 * NSEC_PER_SEC)), dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
    NSLog(@"GCD延时调用(其他线程):%@",[NSThread currentThread]);
});
打印:
[15276:540462] 当前线程 <NSThread: 0x600000068680>{number = 1, name = main}
[15276:540462] GCD延时调用(主线程):<NSThread: 0x600000068680>{number = 1, name = main}
[15276:540521] GCD延时调用(其他线程):<NSThread: 0x60800007c940>{number = 4, name = (null)}

dispatch_once()

在整个程序运行中,代码会以线程安全的方式执行并且只执行一次

for (int i = 0 ; i < 99999; i++) {
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        NSLog(@"执行一次:%d",i);
    });
}

dispatch_barrier_async()

读者写者锁(栅栏函数)

  • 在进程管理中起到一个栅栏的作用,它等待所有位于barrier函数之前的操作执行完毕后执行
  • 在barrier函数执行之后,barrier函数之后的操作才会得到执行
    作用:
    1.实现高效率的数据库访问和文件访问
    2.避免数据竞争

使用:

    dispatch_queue_t queue = dispatch_queue_create("barrier", DISPATCH_QUEUE_CONCURRENT);
dispatch_async(queue, ^{
    NSLog(@"任务1:%@",[NSThread currentThread]);
});

dispatch_async(queue, ^{
    NSLog(@"任务2:%@",[NSThread currentThread]);
});

dispatch_barrier_async(queue, ^{
    NSLog(@"---barrier---:%@",[NSThread currentThread]);
});

dispatch_async(queue, ^{
    NSLog(@"任务3:%@",[NSThread currentThread]);
});

dispatch_async(queue, ^{
    NSLog(@"任务4:%@",[NSThread currentThread]);
});

解决多个网络请求全部发送后再进行操作的问题方案:

1.Dispatch Group

对多个异步任务的完成进行监控
代码中的注释:

  1. 创建一个新的 Dispatch Group,它相当于一个用来记录未完成任务的计数器。
  • dispatch_group_enter,手动通知 Dispatch Group 任务已经开始
  • dispatch_group_leave,手动通知 Dispatch Group 任务已经完成
  • dispatch_group_notify,当 Dispatch Group 中没有任何任务时会执行。

注意:

  • 必须保证 dispatch_group_enter 和 dispatch_group_leave 成对出现,确保进入 Group 的次数和离开 Group 的次数相等。
    - (IBAction)requestForData:(id)sender
    {
        __block NSMutableDictionary *errDict = [NSMutableDictionary dictionaryWithCapacity:0];
        __block NSMutableDictionary *successDic = [NSMutableDictionary dictionaryWithCapacity:0];

        dispatch_group_t requestGroup = dispatch_group_create();//1
    
        for(NSInteger i=1 ; i<4 ;i++)
        {
            dispatch_group_enter(requestGroup);//2
            [self requestForDataWithIndex:i block:^(NSArray *dataArray, NSInteger index, BOOL isSuccess) {
                if (isSuccess) {
                    NSLog(@"第%ld个网络请求成功,返回参数是:%@",(long)index,dataArray);
                    [successDic setObject:dataArray forKey:[NSNumber numberWithInteger:index]];
                }else{
                    NSLog(@"第%ld个网络请求失败,返回参数是:%@",(long)index,dataArray);
                    [errDict setObject:dataArray forKey:[NSNumber numberWithInteger:index]];
                }
                dispatch_group_leave(requestGroup);//3
            }];
        }
        dispatch_group_notify(requestGroup, dispatch_get_main_queue(), ^{//4
            //请求完成,主线程操作
            NSLog(@"请求全部完成,成功数据:%@,失败数据:%@",successDic,errDict);
        });
    }
打印
2017-03-21 14:11:42.098 GCD-Demo[15276:540462] 发起第1个网络请求:<NSThread: 0x600000068680>{number = 1, name = main}
2017-03-21 14:11:42.098 GCD-Demo[15276:540462] 发起第2个网络请求:<NSThread: 0x600000068680>{number = 1, name = main}
2017-03-21 14:11:42.099 GCD-Demo[15276:540462] 发起第3个网络请求:<NSThread: 0x600000068680>{number = 1, name = main}
2017-03-21 14:11:43.190 GCD-Demo[15276:540462] 第1个网络请求成功,返回参数是:(
1
)
2017-03-21 14:11:43.599 GCD-Demo[15276:540462] 第3个网络请求成功,返回参数是:(
1
)
2017-03-21 14:11:44.599 GCD-Demo[15276:540462] 第2个网络请求失败,返回参数是:(
0
)
2017-03-21 14:11:44.599 GCD-Demo[15276:540462] 请求全部完成,成功数据:{
    3 =     (
        1
    );
    1 =     (
        1
    );
},失败数据:{
    2 =     (
        0
    );
}

2.dispatch_apply()

提交队列,适用于并发循环.

- (IBAction)requestForData_Dispatch_apply:(id)sender
{
/**
 dispatch_apply() 

 @param iterations 迭代的次数
 @param queue 指定任务运行的队列
 @param size_t Block
 */
//    dispatch_apply(size_t iterations, dispatch_queue_t  _Nonnull queue, ^(size_t) {})

    __block NSMutableDictionary *errDict = [NSMutableDictionary dictionaryWithCapacity:0];
    __block NSMutableDictionary *successDic = [NSMutableDictionary dictionaryWithCapacity:0];
    dispatch_group_t requestGroup = dispatch_group_create();
    dispatch_apply(3, dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^(size_t i) {
        dispatch_group_enter(requestGroup);
        [self requestForDataWithIndex:i block:^(NSArray *dataArray, NSInteger index, BOOL isSuccess) {
            if (isSuccess) {
                NSLog(@"第%ld个网络请求成功,返回参数是:%@",(long)index,dataArray);
                [successDic setObject:dataArray forKey:[NSNumber numberWithInteger:index]];
            }
            else
            {
                NSLog(@"第%ld个网络请求失败,返回参数是:%@",(long)index,dataArray);
                [errDict setObject:dataArray forKey:[NSNumber numberWithInteger:index]];
            }
            dispatch_group_leave(requestGroup);
        }];
    });
    dispatch_group_notify(requestGroup, dispatch_get_main_queue(), ^{
        //请求完成,主线程操作
        NSLog(@"请求全部完成,成功数据:%@,失败数据:%@",successDic,errDict);
    })
}
打印
2017-03-21 14:15:36.897 GCD-Demo[15276:540462] 发起第0个网络请求:<NSThread: 0x600000068680>{number = 1, name = main}
2017-03-21 14:15:36.897 GCD-Demo[15276:540521] 发起第1个网络请求:<NSThread: 0x60800007c940>{number = 4, name = (null)}
2017-03-21 14:15:36.897 GCD-Demo[15276:612863] 发起第2个网络请求:<NSThread: 0x600000260400>{number = 5, name = (null)}
2017-03-21 14:15:37.993 GCD-Demo[15276:540462] 第1个网络请求成功,返回参数是:(
1
)
2017-03-21 14:15:38.897 GCD-Demo[15276:540462] 第0个网络请求成功,返回参数是:(
1
)
2017-03-21 14:15:39.398 GCD-Demo[15276:540462] 第2个网络请求失败,返回参数是:(
0
)
2017-03-21 14:15:39.398 GCD-Demo[15276:540462] 请求全部完成,成功数据:{
0 =     (
    1
);
1 =     (
    1
);
},失败数据:{
2 =     (
    0
);
}

3.Dispatch Semaphore 信号量

  1. 信号量为0时 会阻塞线程,一直等待
  • dispatch_semaphore_wait(信号量,等待时间) 这个函数会使传入的信号量的值-1;
  • dispatch_semaphore_signal (信号量) 这个函数会使传入的信号量的值+1;
  • 正常的使用顺序是先降低然后再提高,这两个函数通常成对使用。

信号量的理解:

  • 信号量相当于一个停车场,创建时的参数相当于提供多少个车位,如果你有两个车位,有4辆车要停,那么,只能让先进来的两个车子停下,后面的两个车子等待,开走一个,才能停入下一个.dispatch_semaphore_wait函数就相当于来了一辆车,调用一次,车位就-1.dispatch_semaphore_signal函数相当于走了一辆车,调用一次,车位就+1.
- (IBAction)requestForData_Dispatch_semaphore_t:(id)sender
{   
    __block NSMutableDictionary *errDict = [NSMutableDictionary dictionaryWithCapacity:0];
    __block NSMutableDictionary *successDic = [NSMutableDictionary dictionaryWithCapacity:0];

    //创建一个信号量。参数指定信号量的起始值(必须大于0)
    dispatch_semaphore_t semaphore = dispatch_semaphore_create(2);
    for (NSInteger i=0; i<4; i++) {
    
        dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
        
            dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER);//信号量-1
            [self requestForDataWithIndex:i block:^(NSArray *dataArray, NSInteger index, BOOL isSuccess) {
                if (isSuccess) {
                    NSLog(@"第%ld个网络请求成功,返回参数是:%@",(long)index,dataArray);
                    [successDic setObject:dataArray forKey:[NSNumber numberWithInteger:index]];
                }
                else
                {
                    NSLog(@"第%ld个网络请求失败,返回参数是:%@",(long)index,dataArray);
                    [errDict setObject:dataArray forKey:[NSNumber numberWithInteger:index]];
                }
                dispatch_semaphore_signal(semaphore);//信号量+1
            }];
        
        });
    }
}
打印:

这里我代码中设置信号量的起始值为2 ,就会同时发起0和1两个任务.0任务耗时长,1任务先返回后立即发起了任务2,0任务结束后立即发起了任务3.

2017-03-21 14:18:11.150 GCD-Demo[15276:613982] 发起第0个网络请求:<NSThread: 0x60800007fe80>{number = 6, name = (null)}
2017-03-21 14:18:11.150 GCD-Demo[15276:612860] 发起第1个网络请求:<NSThread: 0x608000064280>{number = 7, name = (null)}
2017-03-21 14:18:12.249 GCD-Demo[15276:540462] 第1个网络请求成功,返回参数是:(
1
)
2017-03-21 14:18:12.249 GCD-Demo[15276:613984] 发起第2个网络请求:<NSThread: 0x600000078880>{number = 8, name = (null)}
2017-03-21 14:18:13.151 GCD-Demo[15276:540462] 第0个网络请求成功,返回参数是:(
1
)
2017-03-21 14:18:13.151 GCD-Demo[15276:613985] 发起第3个网络请求:<NSThread: 0x600000077380>{number = 9, name = (null)}
2017-03-21 14:18:14.652 GCD-Demo[15276:540462] 第3个网络请求成功,返回参数是:(
1
)
2017-03-21 14:18:14.749 GCD-Demo[15276:540462] 第2个网络请求失败,返回参数是:(
0
)
  • 注: 这里只是为了理解信号量而提供的一种思路...看看就好,不要抬杠...

最后,附上我用来模拟网络请求的代码:

//模拟网络请求
- (void)requestForDataWithIndex:(NSInteger)index block:(Complete)callback
{
    NSLog(@"发起第%ld个网络请求:%@",(long)index,[NSThread currentThread]);
    NSArray * successArray = [NSArray arrayWithObjects:@1, nil];
    NSArray * failureArray = [NSArray arrayWithObjects:@0, nil];
    if (index == 0) {
        dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(2 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
            if (callback) {
                callback(successArray,index,YES);
            }
        });
    }
    else if (index == 1) {
        dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(1 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
            if (callback) {
                callback(successArray,index,YES);
            }
        });
    }
    else if (index == 2)
    {
        dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(2.5 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
            if (callback) {
                callback(failureArray,index,NO);
            }
        });
    }
    else if (index == 3)
    {
        dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(1.5 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
            if (callback) {
                callback(successArray,index,YES);
            }
        });
    }

}

另附上demo链接

Demo在这里下载...
路漫漫其修远兮...

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

推荐阅读更多精彩内容

  • GCD笔记 总结一下多线程部分,最强大的无疑是GCD,那么先从这一块部分讲起. Dispatch Queue的种类...
    jins_1990阅读 757评论 0 1
  • 背景 担心了两周的我终于轮到去医院做胃镜检查了!去的时候我都想好了最坏的可能(胃癌),之前在网上查的症状都很相似。...
    Dely阅读 9,229评论 21 42
  • 闺蜜从大洋彼岸回来,说是要尽尽孝心带老爹出去玩玩,正好我有空一同前往,途中要去一个海边小城,想起有个同学在那儿,就...
    黄小闲阅读 266评论 2 2
  • 其实相对于喝咖啡 更爱闻咖啡;相对于闻咖啡 更爱去咖啡馆。怎么说呢?!就是 在那个氛围中 我愿意独处 但是有陪伴的...
    Vivian_dh阅读 383评论 0 2
  • 对你 我确实要结束我的幻想了 或许我们根本就不是一路人吧 再见了 这次是真的 我重新找一条出路了 新的一天,新的你...
    chde我阅读 110评论 0 0