深入浅出 GCD 线程使用

串行与并行

同步和异步针对的是线程队列,所谓的线程队列可以理解为一组线程的数组。

串行队列:
队列中是事件有序执行,遵循 FIFO(first in first out)的原则,先进入队列的事件先执行。

串行队列创建:

dispatch_queue_t queue = dispatch_queue_create("com.queue.serial", DISPATCH_QUEUE_SERIAL);

dispatch_get_main_queue() // 主队列,也是串行队列

并行队列
并行队列中的事件在逻辑上是一起执行的,但是这是要根据机器 CPU 的情况而定,在 C++ 线程库中,std::thread::hardware_concurrency() 能获取到当前机器最大能并发的线程数量,iPhone6P 中为 2,也就是说最大同时能处理两个并发线程任务,其他后面添加的任务都得等待两个任务中的其中一个执行完了,才可以执行。

dispatch_queue_t queue = dispatch_queue_create("com.queue.concurrent", DISPATCH_QUEUE_CONCURRENT);

dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0); // 全局并发队列

同步和异步

同步和异步针对的是线程,那么什么是同步线程,什么是异步线程。

同步线程:
阻塞当前线程,要等待同步线程内的任务执行完了并且返回以后,才可以继续执行被阻塞线程的事件。

同步线程创建:

dispatch_sync(queue, block);

异步线程:
不阻塞当前线程,等当前线程完成时间片(完成当前事件)切换后再执行异步线程。

异步线程创建:

dispatch_async(queue, block);

线程问题

主线程中的死锁
NSLog(@"1");
dispatch_sync(dispatch_get_main_queue(), ^(){
    NSLog(@"2");
});
NSLog(@"3");

输出:1

如果上面代码是在主线程当中执行的,那么就会造成我们的死锁问题,注意是主线程当中,后面我们还有一个测试说明。
假定上面代码为主线程中执行的代码,如果不造成死锁的情况是输出应该是 1,2,3,但现在事件只执行了 1,那么死锁就很明显了,我们现在对它进行分析。

dispatch_sync 同步线程,将当前线程阻塞,先执行block(@"2") 然后解放线程
dispatch_get_main_queue 主线程队列,也可以叫做串行队列,将 dispatch_sync 同步线程放到队列后,先执行 ( @"3") 再执行同步线程,遵循 FIFO 的原则。
当时因为 dispatch_sync 是在主线程创建的,所以主线程被阻塞,主线程的事件(@"3") 要等待 dispatch_sync 的 block 执行完后才能执行
所以事件(@"3")无法执行,事件(@"2")更无法执行,相互等待造成死锁。

dispatch_sync(dispatch_get_main_queue(), block)是否一定会造成死锁呢?上面问题如果并不是放在主线程中有会怎么样?

NSLog(@"1");
dispatch_queue_t queue = dispatch_queue_create("com.queue.concurrent", DISPATCH_QUEUE_CONCURRENT);
dispatch_async(queue), ^(){
    NSLog(@"2");
    dispatch_sync(dispatch_get_main_queue(), ^(){
        NSLog(@"3");
    });
    NSLog(@"4");
});
NSLog(@"5");

输出: 1,5,2,3,4

输出中,可以看得出所有事件全部都执行完成,没有造成死锁,但是明明使用了 dispatch_sync(dispatch_get_main_queue(), block);这个经常被说成会造成死锁的方法,但是为什么这里没有造成死锁呢,我们来分析一下。

dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT,block) 中, dispatch_async 异步线程,将其放在了 dispatch_get_global_queue 全局队列,也可以叫并行队列中,主线程不用等待异步 dispatch_async 内的事件(block)执行完成,所以直接执行了事件(@“1”)和事件(@"5")。当线程时间片切换出来,异步线程内的事件(block)便开始执行了,所以事件(@"2") 便执行了。
当运行到 dispatch_sync(dispatch_get_main_queue(),block) 中,dispatch_sync 阻塞当前线程,细想一下,当前线程是一个异步线程并不是主线程,事件(@"4")又是在这个异步线程中的事件,所以要等待 dispatch_sync 同步线程内的事件执行完了,才可以执行。同步线程放在 dispatch_get_main_queue 主线程队列中,主线程队列同时也是一个串行队列,所以事件(@"3") 一定会在事件@("1")和事件(@"5")之后,当执行完事件(@"3")便可以执行事件(@"4")了。

上面例子说明一件事,dispatch_async 同步线程会阻塞当前线程直至同步线程内的事件(block)执行完,至于是否会发生死锁,就得看同步线程所阻塞的线程是否存在它的线程队列(queue)中

current thread

dispatch_sync(queue), block)

第一个例子中,current thread 为主线程,queue 主线程队列,主线程属于主线程队列,所以造成死锁。

第二个例子中,current thread 为我们所开启的异步线程 dispatch_async,并且放在我们自己所创建的 dispatch_queue_t queue = dispatch_queue_create("com.queue.concurrent", DISPATCH_QUEUE_CONCURRENT); 异步线程队列中,queue 为主线程队列,异步线程 dispatch_async 并不属于主线程队列中,所以并没有造成死锁。

异步串行队列和同步串行队列

首先我们做一个比较,在串行队列中开启一个异步线程,然后再异步线程的事件中再开启一个同步线程。(默认下面例子都是在主线程中运行)

dispatch_queue_t queue = dispatch_queue_create("com.queue.CONCURRENT", DISPATCH_QUEUE_CONCURRENT);

NSLog(@"1");
dispatch_async(queue, ^() {
    NSLog(@"2");
    dispatch_sync(queue, ^(){
      NSLog(@"3");
    });
    NSLog(@"4");
});

NSLog(@"5");

输出:1,5,2,3,4

然后将 queue 换成一个串行队列,看看效果如何

dispatch_queue_t queue2 = dispatch_queue_create("com.queue.SERIAL", DISPATCH_QUEUE_SERIAL);

NSLog(@"1");
dispatch_async(queue2, ^() {
    NSLog(@"2");
    dispatch_sync(queue2, ^(){
      NSLog(@"3");
    });
    NSLog(@"4");
});

NSLog(@"5");

输出:1,5,2

第一个例子使用 DISPATCH_QUEUE_CONCURRENT 并发队列,输出正常,而第二个例子中使用了 DISPATCH_QUEUE_SERIAL 串行队列,发生了死锁,后面的事件 (@"3") 和事件 (@"4")便无法执行。

我们首先分析一下第一个例子,为什么并没有发生死锁,首先我们往并发队列 queue 中添加了dispatch_async 异步线程 ,主线程并不等待异步线程的执行,所以事件 (@"1") 后便马上执行事件 (@"5"),当内核线程空闲,加载并发队列 queue 中的 dispatch_async 异步线程 并执行线程中的事件(block) 的,事件 (@"2") 马上就会被执行。
当遇到了 dispatch_sync 同步线程的时候,当前线程,也就是 dispatch_async 这个异步线程会进入阻塞,等待 dispatch_sync 同步线程内的事件(block) 执行完,才可以往下执行事件(@"4"),我们并将dispatch_sync 同步线程放进了 queue 并发队列当中去,并发队列的特点就是逻辑上是一起执行的,所以 dispatch_sync 同步线程加入 queue 后就马上被执行了,当事件(@"3")执行完后并且返回,阻塞放开,事件(@"4")并马上被执行。全过程并没有发生死锁

我们再来看看第二个例子,首先我们往串行队列 queue 中添加了dispatch_async 异步线程 ,其后过程跟第一个例子一样,直到遇到了 dispatch_sync(queue2, block)dispatch_sync` 同步线程 阻塞了 dispatch_async 异步线程,并将同步线程放进了 queue2 串行队列中,串行队列的特别是遵循 FIFO 特点,要必先执行完 dispatch_async 异步线程的事件(block),才能执行同步线程 dispatch_sync 的事件 (block),所以造成了死锁

AFNetWorking 怎么使用同步线程
self.synchronizationQueue = dispatch_queue_create([name cStringUsingEncoding:NSASCIIStringEncoding], DISPATCH_QUEUE_SERIAL);

- (nullable AFImageDownloadReceipt *)downloadImageForURLRequest:(NSURLRequest *)request
                                                  withReceiptID:(nonnull NSUUID *)receiptID
                                                        success:(nullable void (^)(NSURLRequest *request, NSHTTPURLResponse  * _Nullable response, UIImage *responseObject))success
                                                        failure:(nullable void (^)(NSURLRequest *request, NSHTTPURLResponse * _Nullable response, NSError *error))failure {
                                                        
    dispatch_sync(self.synchronizationQueue, ^{
            NSString *URLIdentifier = request.URL.absoluteString;
            if (URLIdentifier == nil) {
                if (failure) {
                    NSError *error;
                    dispatch_async(dispatch_get_main_queue(), ^{
                        failure(request, nil, error);
                    });
                }
                return;
            }
        
            ...
    });
}

上面一段代码才子 AFNetWorking 中的 AFImageDownloader.m 文件当中,作者创建了 synchronizationQueue 串行队列专门用作阻塞当前线程,限制性同步队列中的事件,判断 url 是否为空,但是为什么要这样做呢?

原因:
因为对象方法 downloadImageForURLRequest:withReceiptID:success:failure 是同一个对象在多个异步线程的并发队列当中执行的,因为并发在逻辑上会同时触发异步线程,那么传进来的参数(request,receiptID,success,failure)会由于资源竞争(condition race) 的情况下会被覆盖,所以我们需要进行阻塞这个线程,先执行完一个请求后再执行另外一个请求

但是会有人问:为什么么不用 @synchronized (<#lock#>) {} ?
因为我们首先不确定调用对象方法 downloadImageForURLRequest:withReceiptID:success:failure 是否必定在异步线程中被调用,莫名的加锁会消耗资源,当我们使用了 dispatch_sync(self.synchronizationQueue,block) 后,如果主线程当中被调用,也只会忽视这个方法,直接调用 block,因为阻塞主线程,往并不是主线程队列的线程队列中添加事件,是没有意义的。

使用 dispatch_sync(self.synchronizationQueue,block) 需要注意什么问题?
其实上面这么写,是有问题的,当方法 downloadImageForURLRequest:withReceiptID:success:failure 的调用上层,也是dispatch_sync(self.synchronizationQueue,block) 的情况下,就会造成死锁,就像下面一样

dispatch_sync(self.synchronizationQueue, ^(){
    NSLog(@"2");
    dispatch_sync(self.synchronizationQueue, ^(){
      NSLog(@"3");
    });
    NSLog(@"4");
});

dispatch_async(self.synchronizationQueue, ^(){
    NSLog(@"2");
    dispatch_sync(self.synchronizationQueue, ^(){
      NSLog(@"3");
    });
    NSLog(@"4");
});

至于怎么分析,为什么会发生死锁,各位看官,这就留给你们的作业,看了这么多,相信大家也会明白,特别是第二个例子,我们刚讲过,希望大家能在这篇博客中学到东西。


写在最后:

为什么要写这篇文章呢?主要今天在某公司面试的时候,被问到了关于 GCD 的线程问题,在我说出来答案后,面试官依然坚持已见,认为我是错的,让该面试官指出哪里错误的时候,该面试官又在故弄玄虚,并让我错失了这个宝贵的机会,写这篇博客的目的在于,不管这个面试官是否会游览博客,也让更多的面试官可以好好更新自己的知识储备库,不要坐井观天。其实在我看来,面试是一个双向交流的过程,我并不在意是否能你们公司工作,毕竟我也不想与一群无法交流的人共事,一个开心愉快并且能够助我成长的工作环境才是我真正需要的。

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

推荐阅读更多精彩内容