一.什么是GCD

Grand Central Dispatch (强大的中枢调度器) ,是异步执行任务的技术之一。纯C语言,有很多强大的函数。

二.GCD的优势

(1)GCD是苹果公司为多核并行运算提供的解决方案。

(2)GCD会自动利用更多的CPU内核(双核,四核等)。

(3)GCD会自动管理线程的生命周期(创建线程、调度任务、销毁线程)

   (4)程序员只需要告诉GCD想要执行什么任务,不需要编写任何线程管理代码

 三.任务和队列

  GCD中有两个很核心的概念:任务和队列。

  任务:你要执行的操作

队列:用来存放任务

四.执行任务的两种方式

      1.同步:在当前线程中执行

  异步:在另一个线程中执行

  2. GCD中有两个用来执行任务的函数:

   (1)用同步的方式执行任务 dispatch_sync(dispatch_queue_t queue, dispatch_block_t block);

    (2)用异步的方式执行任务 dispatch_async(dispatch_queue_t queue, dispatch_block_t block);

五.队列的两种类型

   (1)  串行队列:在一个线程中依次执行(dispatch_sync)

     并发队列:多个任务同时执行 (开启多个线程同时执行任务)(dispatch_async)

(2)同步、异步、串行、并发 说明

     同步和异步决定了能不能开启新线程

     同步:在当前线程中执行任务,不具备开启新线程的能力

     异步:在新的线程中执行任务,具备开启新线程的能力  

    串行和并行决定了任务的执行形式

    串行:任务依次执行

    并行:多个任务同时并发执行

   注意:只有在异步的时候开启线程,所以只能在异步的时候并发执行任务。

六.获取队列的函数

1获取串行队列 

(1)使用dispatch_queue_create函数创建串行队列

dispatch_queue_t queue = dispatch_queue_create("wendingding", NULL); // 创建

(2)使用主队列(跟主线程相关联的队列)

dispatch_queue_t queue = dispatch_get_main_queue();

2.获取并行队列 

使用dispatch_get_global_queue函数获得全局的并发队列

dispatch_queue_t dispatch_get_global_queue(dispatch_queue_priority_t priority,unsigned long flags); // 此参数暂时无用,用0即可

dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, ); // 获得全局并发队列

说明:全局并发队列的优先级

#define DISPATCH_QUEUE_PRIORITY_HIGH 2 // 高

#define DISPATCH_QUEUE_PRIORITY_DEFAULT 0 // 默认(中)

#define DISPATCH_QUEUE_PRIORITY_LOW (-2) // 低

#define DISPATCH_QUEUE_PRIORITY_BACKGROUND INT16_MIN // 后台

七、GCD的基本使用

1. 串行队列 + 同步执行

- (void)viewDidLoad
{
[super viewDidLoad];
NSLog(@"主线程----%@",[NSThread mainThread]); //创建串行队列
dispatch_queue_t queue = dispatch_queue_create("dengdeng", ); //添加任务到队列中去
dispatch_sync(queue, ^{
NSLog(@"下载图片1 ----- %@",[NSThread currentThread]);
}); dispatch_sync(queue, ^{
NSLog(@"下载图片2 ----- %@",[NSThread currentThread]);
}); dispatch_sync(queue, ^{
NSLog(@"下载图片3 ----- %@",[NSThread currentThread]);
}); }

打印结果如下:

2017-06-16 16:56:16.331 GCDTest[14678:2213850] 主线程----<NSThread: 0x608000073000>{number = 1, name = main}

2017-06-16 16:56:16.332 GCDTest[14678:2213850] 下载图片1 ----- <NSThread: 0x608000073000>{number = 1, name = main}

2017-06-16 16:56:16.333 GCDTest[14678:2213850] 下载图片2 ----- <NSThread: 0x608000073000>{number = 1, name = main}

2017-06-16 16:56:16.333 GCDTest[14678:2213850] 下载图片3 ----- <NSThread: 0x608000073000>{number = 1, name = main}

2. 并行队列 + 同步执行

- (void)viewDidLoad

{

[super viewDidLoad];

NSLog(@"主线程----%@",[NSThread mainThread]);

//创建串行队列

dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);

//第一个参数为串行队列的名称,是c语言的字符串

//第二个参数为队列的属性,一般来说串行队列不需要赋值任何属性,所以通常传空值(NULL)

//添加任务到队列中去

dispatch_sync(queue, ^{

for (int i = 0; i < 2 ; i ++) {

NSLog(@"下载图片1 ----- %@",[NSThread currentThread]);

}

});

dispatch_sync(queue, ^{

for (int i = 0; i < 2 ; i ++) {

NSLog(@"下载图片2 ----- %@",[NSThread currentThread]);

}

});

dispatch_sync(queue, ^{

for (int i = 0; i < 2 ; i ++) {

NSLog(@"下载图片3 ----- %@",[NSThread currentThread]);

}

});

}

打印结果:

2017-06-16 17:29:58.848 GCDTest[14951:2249589] 主线程----<NSThread: 0x600000068600>{number = 1, name = main}

2017-06-16 17:29:58.849 GCDTest[14951:2249589] 下载图片1 ----- <NSThread: 0x600000068600>{number = 1, name = main}

2017-06-16 17:29:58.849 GCDTest[14951:2249589] 下载图片1 ----- <NSThread: 0x600000068600>{number = 1, name = main}

2017-06-16 17:29:58.849 GCDTest[14951:2249589] 下载图片2 ----- <NSThread: 0x600000068600>{number = 1, name = main}

2017-06-16 17:29:58.849 GCDTest[14951:2249589] 下载图片2 ----- <NSThread: 0x600000068600>{number = 1, name = main}

2017-06-16 17:29:58.849 GCDTest[14951:2249589] 下载图片3 ----- <NSThread: 0x600000068600>{number = 1, name = main}

2017-06-16 17:29:58.849 GCDTest[14951:2249589] 下载图片3 ----- <NSThread: 0x600000068600>{number = 1, name = main}

注:同步执行不会开启新线程

2. 串行队列 + 异步执行

- (void)viewDidLoad
{
[super viewDidLoad];
NSLog(@"主线程----%@",[NSThread mainThread]);
//创建串行队列
dispatch_queue_t queue = dispatch_queue_create("dengdeng", );
//第一个参数为串行队列的名称,是c语言的字符串
//第二个参数为队列的属性,一般来说串行队列不需要赋值任何属性,所以通常传空值(NULL) //添加任务到队列中去
dispatch_async(queue, ^{
NSLog(@"下载图片1 ----- %@",[NSThread currentThread]);
}); dispatch_async(queue, ^{
NSLog(@"下载图片2 ----- %@",[NSThread currentThread]);
}); dispatch_async(queue, ^{
NSLog(@"下载图片3 ----- %@",[NSThread currentThread]);
}); }

2017-06-16 17:15:09.901 GCDTest[14851:2235809] 主线程----<NSThread: 0x60800007ad80>{number = 1, name = main}

2017-06-16 17:15:09.901 GCDTest[14851:2235848] 下载图片1 ----- <NSThread: 0x608000262c40>{number = 3, name = (null)}

2017-06-16 17:15:09.901 GCDTest[14851:2235848] 下载图片2 ----- <NSThread: 0x608000262c40>{number = 3, name = (null)}

2017-06-16 17:15:09.902 GCDTest[14851:2235848] 下载图片3 ----- <NSThread: 0x608000262c40>{number = 3, name = (null)}

注:开启了新的线程,在新的线程上,这三个任务是串行的。

- (void)viewDidLoad
{
[super viewDidLoad];
NSLog(@"主线程----%@",[NSThread mainThread]); //创建串行队列
dispatch_queue_t queue = dispatch_queue_create("dengdeng", );
//第一个参数为串行队列的名称,是c语言的字符串
//第二个参数为队列的属性,一般来说串行队列不需要赋值任何属性,所以通常传空值(NULL)
//添加任务到队列中去
dispatch_async(queue, ^{
NSLog(@"下载图片1 ----- %@",[NSThread currentThread]);
}); dispatch_async(queue, ^{
NSLog(@"下载图片2 ----- %@",[NSThread currentThread]);
}); dispatch_async(queue, ^{
NSLog(@"下载图片3 ----- %@",[NSThread currentThread]);
}); for (int i = ; i < ; i ++) {
NSLog(@"下载图片main ----- %@",[NSThread currentThread]);
} }

2017-06-16 17:19:06.154 GCDTest[14870:2240144] 主线程----<NSThread: 0x60800007d200>{number = 1, name = main}

2017-06-16 17:19:06.155 GCDTest[14870:2240144] 下载图片main ----- <NSThread: 0x60800007d200>{number = 1, name = main}

2017-06-16 17:19:06.155 GCDTest[14870:2240278] 下载图片1 ----- <NSThread: 0x600000268400>{number = 3, name = (null)}

2017-06-16 17:19:06.155 GCDTest[14870:2240144] 下载图片main ----- <NSThread: 0x60800007d200>{number = 1, name = main}

2017-06-16 17:19:06.155 GCDTest[14870:2240278] 下载图片2 ----- <NSThread: 0x600000268400>{number = 3, name = (null)}

2017-06-16 17:19:06.155 GCDTest[14870:2240144] 下载图片main ----- <NSThread: 0x60800007d200>{number = 1, name = main}

2017-06-16 17:19:06.155 GCDTest[14870:2240278] 下载图片3 ----- <NSThread: 0x600000268400>{number = 3, name = (null)}

注:开启的新线程和主线程的关系:并发执行 

4. 并行队列 + 异步执行

 

- (void)viewDidLoad
{
[super viewDidLoad];
NSLog(@"主线程----%@",[NSThread mainThread]); //创建串行队列
dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, );
//第一个参数为串行队列的名称,是c语言的字符串
//第二个参数为队列的属性,一般来说串行队列不需要赋值任何属性,所以通常传空值(NULL)
//添加任务到队列中去
dispatch_async(queue, ^{
for (int i = ; i < ; i ++) {
NSLog(@"下载图片1 ----- %@",[NSThread currentThread]);
}
}); dispatch_async(queue, ^{
for (int i = ; i < ; i ++) {
NSLog(@"下载图片2 ----- %@",[NSThread currentThread]);
}
}); }

2017-06-16 17:25:40.722 GCDTest[14920:2245705] 主线程----<NSThread: 0x60000007f240>{number = 1, name = main}

2017-06-16 17:25:40.725 GCDTest[14920:2245740] 下载图片1 ----- <NSThread: 0x60000026b9c0>{number = 3, name = (null)}

2017-06-16 17:25:40.725 GCDTest[14920:2245742] 下载图片2 ----- <NSThread: 0x608000264b00>{number = 4, name = (null)}

2017-06-16 17:25:40.752 GCDTest[14920:2245740] 下载图片1 ----- <NSThread: 0x60000026b9c0>{number = 3, name = (null)}

2017-06-16 17:25:40.753 GCDTest[14920:2245742] 下载图片2 ----- <NSThread: 0x608000264b00>{number = 4, name = (null)}

2017-06-16 17:25:40.755 GCDTest[14920:2245740] 下载图片1 ----- <NSThread: 0x60000026b9c0>{number = 3, name = (null)}

2017-06-16 17:25:40.756 GCDTest[14920:2245742] 下载图片2 ----- <NSThread: 0x608000264b00>{number = 4, name = (null)}

注:开启多个线程,并发执行任务

 

iOS —— GCD 详解的更多相关文章

  1. iOS GCD详解

    前言 对初学者来说,GCD似乎是一道迈不过去的坎,很多人在同步.异步.串行.并行和死锁这几个名词的漩涡中渐渐放弃治疗.本文将使用图文表并茂的方式给大家形象地解释其中的原理和规律. 线程.任务和队列的概 ...

  2. 【转】IOS AutoLayout详解(三)用代码实现(附Demo下载)

    转载自:blog.csdn.net/hello_hwc IOS SDK详解 前言: 在开发的过程中,有时候创建View没办法通过Storyboard来进行,又需要AutoLayout,这时候用代码创建 ...

  3. Swift - 多线程GCD详解

    //  GCD详解 //  目录: //  1. 创建GCD队列(最常用) //  2. 自定义创建队列 //  3. 使用多线程实现延迟加载 //  4. 使用多线程实现重复(循环) //  5. ...

  4. IOS SDK详解

    来源:http://blog.csdn.net/column/details/huangwenchen-ios-sdk.html?page=1#42803301 博客专栏>移动开发专栏>I ...

  5. iOS路由详解

    本文如题,路由详解,注定是一篇详细解释iOS路由原理及使用的文章,由于此时正在外地出差,无法详细一一写出,只能不定时的补充. 一.什么是iOS路由 路由一词来源于路由器,可以实现层级之间消息转发的功能 ...

  6. IOS 手势详解

    在IOS中手势可以让用户有很好的体验,因此我们有必要去了解一下手势. (在设置手势是有很多值得注意的地方) *是需要设置为Yes的点击无法响应* *要把手势添加到所需点击的View,否则无法响应* 手 ...

  7. IOS SizeClasses 详解

    SizeClasses 详解 iOS 8在应用界面的可视化设计上添加了一个新的特性-Size Classes.对于任何设备来说,界面的宽度和高度都只分为三种描述:紧凑,任意和宽松.这样开发者便可以无视 ...

  8. iOS模式详解—「runtime面试、工作」看我就 🐒 了 ^_^.

    Write in the first[写在最前] 对于从事 iOS 开发人员来说,当提到 ** runtime时,我想都可以说出来 「runtime 运行时」和基本使用的方法.相信很多开发者跟我当初一 ...

  9. iOS 模式详解—「runtime面试、工作」看我就 🐒 了 ^_^.

    引导 Copyright © PBwaterln Unauthorized shall not be *copy reprinted* . 对于从事 iOS 开发人员来说,所有的人都会答出「runti ...

随机推荐

  1. 换行符在HTML中直接替换为<br>

         #set($text=$!obj.getMeasure().replaceAll("\r\n","<br>"))     <td a ...

  2. 557. Reverse Words in a String III

    static int wing=[]() { std::ios::sync_with_stdio(false); cin.tie(NULL); ; }(); class Solution { publ ...

  3. hdu-1124(数学问题,求n!的尾零的个数)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1124 思路:每五个数1个0,5个5就2个0(不用管2,一定充足) #include<iostre ...

  4. 《Linux多线程编程手册》读书笔记

    第二章 基本线程编程 1.(P25)如果多个线程等待同一个线程终止,则所有等待线程将一直等到目标线程终止.然后,一个等待线程成功返回,其余的等待线程将失败并返回ESRCH错误. 2.(P26)将新线程 ...

  5. chandy-lamport 分布式一致性快照 算法详细介绍

    在一个分布式计算系统中,为了保证数据的一致性需要对数据进行一致性快照.Flink和spark在做流失计算的时候都借鉴了chandy-lamport算法的原理,这篇文章就是对chandy-lamport ...

  6. 使用async-http-client实现异步批量http请求

    最近项目中需要在微服务中调用rest接口,而且需要调用得次数很多,所以同步得http客户端已经不满足要求,在网上查阅资料后发现了async-http-client这个包得性能不错,所以写了个demo测 ...

  7. node.js初步总结

    一:先上一段代码 process.argv.forEach(function (val, index, array) {    console.log(index + ":" + ...

  8. 7系列GTX中的疑惑

    1.PCOMMA与MCOMMA指什么? PCOMMA是指RD-部分的数据,MCOMMA是指RD+部分的数据. 2.risk信号作用? risk信号来指示有效的K码. 如果不选择用8b10b来编码,是需 ...

  9. MEF程序设计指南

    ############################################################################################## ##### ...

  10. virtualenv 安装及使用[转]

    一如既往,官方文档: https://virtualenv.pypa.io/en/latest/   下载地址:https://pypi.python.org/pypi/virtualenv#down ...