iOS多线程GCD简介(一)

之前讲过多线程之NSOperation,今天来讲讲代码更加简洁和高效的GCD。下面说的内容都是基于iOS6以后和ARC下。

Grand Central Dispatch (GCD)简介

Grand Central Dispatch(GCD) 是异步执行任务的技术之一。开发者只需要定义想执行的任务并追加到适当的Dispatch Queue中,GCD就能生成必要的线程并计划执行任务。由于线程管理是作为系统的一部分来实现的,因此可以统一管理,也可执行任务,这样就比以前的线程更有效率。GCD用非常简洁的代码,就可以实现多线程编程。

这篇主要讲Dispatch Queue的一些基本东西,后续会加入其他相关内容的介绍。

Dispatch Queue 种类

Dispatch Queue是执行处理的等待队列,通过调用dispatch_async等函数,以block的形式将任务追加到Dispatch Queue中。Dispatch Queue按照添加进来的顺序(FIFO)执行任务处理。但是在任务执行处理方式上,分为Serial Dispatch QueueConcurrent Dispatch Queue。两者的区别如表格所示

Dispatch Queue分类 说明
Serial Dispatch Queue 串行的队列,每次只能执行一个任务,并且必须等待前一个执行任务完成
Concurrent Dispatch Queue 一次可以并发执行多个任务,不必等待执行中的任务完成

下面用代码来演示下:

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

如果上面的queue是Serial Dispatch Queue的话,那么输出的结果一定是1,2,3,4。因为执行顺序是确定的,并且后续的任务必须在之前的任务执行完成后才能执行。

如果是Concurrent Dispatch Queue的话,那么输出的结果就不一定是1,2,3,4了。因为这些任务都是并发执行,并且不需要等待执行中的任务完成,如果其中任意一个任务完成将立即执行后面的任务。

Dispatch Queue 创建

在自定义创建前,我们先看看系统为我们提供的几个全局的Dispatch Queue:

名称 Dispatch Queue 的种类 说明
Main Dispatch Queue Serial Dispatch Queue 主线程执行
Global Dispatch Queue (HIGH) Concurrent Dispatch Queue 执行优先级:高
Global Dispatch Queue (DEFAULT) Concurrent Dispatch Queue 执行优先级:默认
Global Dispatch Queue (LOW) Concurrent Dispatch Queue 执行优先级:低
Global Dispatch Queue (BACKGROUND) Concurrent Dispatch Queue 执行优先级:后台

从表格中我们可以知道我们的主线程就是Serial Dispatch Queue,而之后的三种Dispatch Queue 则是Concurrent Dispatch Queue。这也是为什么我们不能把耗时的任务放在主线程里面去操作。

如果没有特殊需求,我们可以直接获取这些queue来执行我们的任务:

//主线程
    dispatch_queue_t mainQueue = dispatch_get_main_queue();
//HIGH
    dispatch_queue_t highQueue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_HIGH, 0);
//DEFAULT
    dispatch_queue_t defaultQueue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);
//LOW
    dispatch_queue_t lowQueue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_LOW, 0);
//BACKGROUND
    dispatch_queue_t backgroundQueue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_BACKGROUND, 0);

下面看看如何自定义一个queue:

//串行队列
    dispatch_queue_t serialQueue = dispatch_queue_create("com.gcd.serialQueue", DISPATCH_QUEUE_SERIAL);
//并发队列
    dispatch_queue_t concurrentQueue = dispatch_queue_create("com.gcd.concurrentQueue", DISPATCH_QUEUE_CONCURRENT);

通过调用dispatch_queue_create(const char *label, dispatch_queue_attr_t attr)这个函数。第一个参数是给这个queue起的标识,这个在调试的可以看到是哪个队列在执行,或者在crash日志中,也能做为提示。第二个是需要创建的队列类型,是串行的还是并发的。当然你也可以通过dispatch_queue_get_label(dispatch_queue_t queue)获取你创建queue的名字。

使用

  • 异步执行

这个也是我们使用最多的地方,我们直接调用dispatch_async这个函数,就可以将我们要追加的任务添加到队列里面,并立即返回,异步的执行。这个不多讲。

dispatch_async(queue, ^{
        NSLog(@"1");
    });
  • 同步执行

这点我们可能用得不是很多,但是一用不好就出现问题了。当调用这个dispatch_sync函数的时候,这个线程将不会立即返回,直到这个线程执行完毕。看下下面的代码:

dispatch_sync(queue, ^{
        [NSThread sleepForTimeInterval:3];
        NSLog(@"2");
    });

如果你在主线程里面调用这个函数,那么,很遗憾,主线程将被卡主3秒钟。当主线程调用这个方法的时候,由于是同步,不会立即返回,直到这个里面内容执行完毕才能返回。这个时候不管你这个queue是什么类型,都一样。既然不能立即返回,我们可以在一个异步执行的线程中,再去调用这个同步方法。

dispatch_async(serialQueue, ^{
        NSLog(@"4");
        dispatch_sync(queue, ^{
            [NSThread sleepForTimeInterval:3];
            NSLog(@"5");
        });
        NSLog(@"6");
    });

那这个时候,调用这个方法的时候这个线程将立即返回。而同步在这个线程里面执行。这个时候将输出,4,5,6的结果。

同步执行的死锁问题

现在把上面代码拿出来改下


dispatch_async(serialQueue, ^{
        NSLog(@"4");
        dispatch_sync(serialQueue, ^{
            [NSThread sleepForTimeInterval:3];
            NSLog(@"5");
        });
        NSLog(@"6");
    });

这个时候,我把queue的类型设置为串行的类型。这个时候将只会输出4。为什么呢?系统调用这个线程的时候,首先输出4,然后继续执行这个里面的同步线程。由于我的这个queue是串行的,也就是后续的任务必须在之前的执行中任务完成后才能继续执行。但是这个同步执行的线程不会立即返回,必须等到它执行完成才能返回。这样最外面的任务没法执行完,而里面的同步线程又不能立即返回,所以就形成了死锁。

因此在你的编程中使用这个API的时候,一定要当心,不然就会形成死锁。

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

推荐阅读更多精彩内容

  • 本篇博客共分以下几个模块来介绍GCD的相关内容: 多线程相关概念 多线程编程技术的优缺点比较? GCD中的三种队列...
    有梦想的老伯伯阅读 995评论 0 4
  • iOS 多线程系列 -- 基础概述iOS 多线程系列 -- pthreadiOS 多线程系列 -- NSThrea...
    shannoon阅读 805评论 0 2
  • 程序中同步和异步是什么意思?有什么区别? 解释一:异步调用是通过使用单独的线程执行的。原始线程启动异步调用,异步调...
    风继续吹0阅读 998评论 1 2
  • 目录 一、基本概念1.多线程2.串行和并行, 并发3.队列与任务4.同步与异步5.线程状态6.多线程方案 二、GC...
    BohrIsLay阅读 1,466评论 5 12
  • 从哪说起呢? 单纯讲多线程编程真的不知道从哪下嘴。。 不如我直接引用一个最简单的问题,以这个作为切入点好了 在ma...
    Mr_Baymax阅读 2,676评论 1 17