[New learn]GCD的基本使用
https://github.com/xufeng79x/GCDDemo
1.简介
介绍GCD的使用,介绍多种队列与同步异步多种情况下的组合运行情况。
2.基本使用步骤
如果使用GCD则一般也就两个步骤
1.创建/获取执行队列
2.同步或者异步在队列中执行任务 ,通常任务为一个block
3.队列与执行类型(同步或者异步)的组合运行情况
GCD的队列分为串行队列和并发队列,这两种队里都可以自行创建。
除了能够自行创建队列外,系统中已经存在了主队列和全局队列,这两个队列我们可以直接获取使用。
GCD的任务执行方式有分为同步和异步两种。所以根据队列类型和执行方式我们可以排列组合进行选择,下面我们将对这种组合
加以测试来探索他们的不同表现。
| 执行方式/队列类型 | 自建串行队列 | 自建并发队列 | 主队列 | 全局队列 |
|
同步 |
||||
|
异步 |
3.1 自建串行队列+同步执行
/**
* 串行+同步方式执行任务
* 任务顺序:先进先出
* 线程启动方式:不开启新线程,全部在当前线程中执行
*/
-(void) test1
{
// 创建串行队列
dispatch_queue_t queue = dispatch_queue_create("xf", DISPATCH_QUEUE_SERIAL);
// 循环调用,确定任务的放置于启动顺序
; i < ; i++)
{
// 同步方式启动任务
dispatch_sync(queue,^{
NSLog(@"execute task : %d in thread : %@", i, [NSThread currentThread]);
});
}
NSLog(@"%@", @"完成!");
}
执行结果:
-- :::] execute task : , name = main} -- :::] execute task : , name = main} -- :::] execute task : , name = main} -- :::] execute task : , name = main} -- :::] execute task : , name = main} -- :::] execute task : , name = main} -- :::] execute task : , name = main} -- :::] execute task : , name = main} -- :::] execute task : , name = main} -- :::] execute task : , name = main} -- :::] 完成!
总结:
/** * 串行+同步方式执行任务 * 任务顺序:先进先出 * 线程启动方式:不开启新线程,全部在当前线程中执行 */
3.2 自建串行队列+异步执行
/**
* 串行+异步方式执行任务
* 任务顺序:先进先出
* 线程启动方式:开启新线程且只开启一个新线程,全部在新线程中执行
*/
-(void) test2
{
// 创建串行队列
dispatch_queue_t queue = dispatch_queue_create("xf", DISPATCH_QUEUE_SERIAL);
// 循环调用,确定任务的放置于启动顺序
; i < ; i++)
{
// 同步方式启动任务
dispatch_async(queue,^{
NSLog(@"execute task : %d in thread : %@", i, [NSThread currentThread]);
});
}
NSLog(@"%@", @"完成!");
}
结果:
-- :::] 完成! -- :::] execute task : , name = (null)} -- :::] execute task : , name = (null)} -- :::] execute task : , name = (null)} -- :::] execute task : , name = (null)} -- :::] execute task : , name = (null)} -- :::] execute task : , name = (null)} -- :::] execute task : , name = (null)} -- :::] execute task : , name = (null)} -- :::] execute task : , name = (null)} -- :::] execute task : , name = (null)}
总结:
/** * 串行+异步方式执行任务 * 任务顺序:先进先出 * 线程启动方式:开启新线程且只开启一个新线程,全部在新线程中执行 */
3.3 自建并发队列+同步执行
/**
* 并发队列+同步方式执行任务
* 任务顺序:先进先出
* 线程启动方式:不开启新线程,全部在当前线程中执行
*/
-(void) test3
{
// 创建队列
dispatch_queue_t queue = dispatch_queue_create("xf", DISPATCH_QUEUE_CONCURRENT);
// 循环调用,确定任务的放置于启动顺序
; i < ; i++)
{
// 启动任务
dispatch_sync(queue,^{
NSLog(@"execute task : %d in thread : %@", i, [NSThread currentThread]);
});
}
NSLog(@"%@", @"完成!");
}
结果:
-- :::] execute task : , name = main} -- :::] execute task : , name = main} -- :::] execute task : , name = main} -- :::] execute task : , name = main} -- :::] execute task : , name = main} -- :::] execute task : , name = main} -- :::] execute task : , name = main} -- :::] execute task : , name = main} -- :::] execute task : , name = main} -- :::] execute task : , name = main} -- :::] 完成!
总结:
/** * 并发队列+同步方式执行任务 * 任务顺序:先进先出 * 线程启动方式:不开启新线程,全部在当前线程中执行 */
3.4 自建并发队列+异步执行
/**
* 并发队列+异步方式执行任务
* 任务顺序:无序
* 线程启动方式:开启多个线程,在多个线程中可以同时执行多个任务
*/
-(void) test4
{
// 创建队列
dispatch_queue_t queue = dispatch_queue_create("xf", DISPATCH_QUEUE_CONCURRENT);
// 循环调用,确定任务的放置于启动顺序
; i < ; i++)
{
// 启动任务
dispatch_async(queue,^{
NSLog(@"execute task : %d in thread : %@", i, [NSThread currentThread]);
});
}
NSLog(@"%@", @"完成!");
}
结果:
-- :::] 完成! -- :::] execute task : , name = (null)} -- :::] execute task : , name = (null)} -- :::] execute task : , name = (null)} -- :::] execute task : , name = (null)} -- :::] execute task : , name = (null)} -- :::] execute task : , name = (null)} -- :::] execute task : , name = (null)} -- :::] execute task : , name = (null)} -- :::] execute task : , name = (null)} -- :::] execute task : , name = (null)}
总结:
/** * 并发队列+异步方式执行任务 * 任务顺序:无序 * 线程启动方式:开启多个线程,在多个线程中可以同时执行多个任务 */
接下来我们来测试一下两个特殊的队列:主队列和全局队列
主队列:负责在主线程上调度任务,顺序执行其中任务。其中的任务不会运行与子线程,换句话说无论是同步还是异步都不会开子线程。
3.5 主队列+同步执行
/**
* 主队列+同步方式执行任务
* 任务顺序:卡死
* 线程启动方式:卡死
*/
-(void) test5
{
// 创建队列
dispatch_queue_t queue = dispatch_get_main_queue();
// 循环调用,确定任务的放置于启动顺序
; i < ; i++)
{
// 启动任务
dispatch_sync(queue,^{
NSLog(@"execute task : %d in thread : %@", i, [NSThread currentThread]);
});
}
NSLog(@"%@", @"完成!");
}
结果:
无
总结:
/** * 主队列+同步方式执行任务 * 任务顺序:卡死 * 线程启动方式:卡死 */
3.6 主队列+异步执行
/** * 主队列+异步方式执行任务 * 任务顺序:先进先出 * 线程启动方式:不开启新线程,只在主线程中执行(需要等到主线程中当前任务完成后再执行后续任务,可观察到先打印“完成“) */
-(void) test6
{
// 创建队列
dispatch_queue_t queue = dispatch_get_main_queue();
// 循环调用,确定任务的放置于启动顺序
; i < ; i++)
{
// 启动任务
dispatch_async(queue,^{
NSLog(@"execute task : %d in thread : %@", i, [NSThread currentThread]);
});
}
NSLog(@"%@", @"完成!");
}
结果:
-- :::] 完成! -- :::] execute task : , name = main} -- :::] execute task : , name = main} -- :::] execute task : , name = main} -- :::] execute task : , name = main} -- :::] execute task : , name = main} -- :::] execute task : , name = main} -- :::] execute task : , name = main} -- :::] execute task : , name = main} -- :::] execute task : , name = main} -- :::] execute task : , name = main}
总结:
/** * 主队列+异步方式执行任务 * 任务顺序:先进先出 * 线程启动方式:不开启新线程,只在主线程中执行(需要等到主线程中当前任务完成后再执行后续任务,可观察到先打印“完成”) */
3.7 全局队列+同步执行
/**
* 全局队列+同步方式执行任务
* 任务顺序:先进先出
* 线程启动方式:不开启新线程,只在主线程中执行
*/
-(void) test7
{
// 创建队列
dispatch_queue_t queue = dispatch_get_global_queue(, );
// 循环调用,确定任务的放置于启动顺序
; i < ; i++)
{
// 启动任务
dispatch_sync(queue,^{
NSLog(@"execute task : %d in thread : %@", i, [NSThread currentThread]);
});
}
NSLog(@"%@", @"完成!");
}
结果:
-- :::] execute task : , name = main} -- :::] execute task : , name = main} -- :::] execute task : , name = main} -- :::] execute task : , name = main} -- :::] execute task : , name = main} -- :::] execute task : , name = main} -- :::] execute task : , name = main} -- :::] execute task : , name = main} -- :::] execute task : , name = main} -- :::] execute task : , name = main} -- :::] 完成!
总结:
/** * 全局队列+同步方式执行任务 * 任务顺序:先进先出 * 线程启动方式:不开启新线程,只在主线程中执行 */
3.8 全局队列+异步执行
/**
* 全局队列+异步方式执行任务
* 任务顺序:无序
* 线程启动方式:开启多个新线程,不同任务可在不同线程中并发执行
*/
-(void) test8
{
// 创建队列
dispatch_queue_t queue = dispatch_get_global_queue(, );
// 循环调用,确定任务的放置于启动顺序
; i < ; i++)
{
// 启动任务
dispatch_async(queue,^{
NSLog(@"execute task : %d in thread : %@", i, [NSThread currentThread]);
});
}
NSLog(@"%@", @"完成!");
}
结果:
-- :::] 完成! -- :::] execute task : , name = (null)} -- :::] execute task : , name = (null)} -- :::] execute task : , name = (null)} -- :::] execute task : , name = (null)} -- :::] execute task : , name = (null)} -- :::] execute task : , name = (null)} -- :::] execute task : , name = (null)} -- :::] execute task : , name = (null)} -- :::] execute task : , name = (null)} -- :::] execute task : , name = (null)}
总结:
/** * 全局队列+异步方式执行任务 * 任务顺序:无序 * 线程启动方式:开启多个新线程,不同任务可在不同线程中并发执行 */
4.至此对于GCD的一般用法总结如下:
| 执行方式/队列类型 | 自建串行队列 | 自建并发队列 | 主队列 | 全局队列 |
|
同步 |
1.不开启新线程 2.先进先出顺序执行 |
1.不开启新线程 2.先进先出顺序执行 |
卡死 |
1.不开启新线程 2.先进先出顺序执行 |
|
异步 |
1.开启新线程,且只开启一个新线程。 2.先进先出顺序执行,且只在新线程中执行 |
1.开启多个线程,任务可在多个线程中并发执行 2.无序执行 |
1.不开启新线程,只在主线程中执行 2.先进先出顺序执行 |
1.开启多个线程,任务可在多个线程中并发执行 2.无序执行 |
有此我们可以看出:
1.同步执行方式都不会新建线程。
2.当队列为串行队列,则在异步执行方式下也只会开启一个线程,因为队列性质决定,开启多个线程也是徒劳无功。
3.主队列只能在主线程下执行。
3.全局队列的表现和自建并发队列一致,所以平时可以直接使用全局队列。
5.更进一步
我们看到GCD中存在卡死的现象,那么什么时候回出现这种情况呢?下一篇博文[New learn]GCD的卡死现象分析研究继续分析研究。
[New learn]GCD的基本使用的更多相关文章
- [New learn]GCD的卡死现象分析研究
https://github.com/xufeng79x/GCDDemo 1.简介 前接[New learn]GCD的基本使用,我们分析了GCD的一般使用方法,其中比较特殊的是在分析到主队列的时候发生 ...
- [New learn]GCD其他方法的使用
https://github.com/xufeng79x/GCDDemo 1.简介 在前面的两篇博文中我介绍了GCD的一般使用方法和死锁的分析调查.本博文中继续讲解GCD的其他比较常用的几个使用方法. ...
- bzoj 2818: Gcd GCD(a,b) = 素数
2818: Gcd Time Limit: 10 Sec Memory Limit: 256 MBSubmit: 1566 Solved: 691[Submit][Status] Descript ...
- Multithreading annd Grand Central Dispatch on ios for Beginners Tutorial-多线程和GCD的入门教程
原文链接:Multithreading and Grand Central Dispatch on iOS for Beginners Tutorial Have you ever written a ...
- SPOJ PGCD 4491. Primes in GCD Table && BZOJ 2820 YY的GCD (莫比乌斯反演)
4491. Primes in GCD Table Problem code: PGCD Johnny has created a table which encodes the results of ...
- [New learn] 设计模式思考
本文是对上文[New learn] 设计模式的思考总结 1.大框架 无论应用使用多少种设计模式和技巧,此模式都是应用的大框架.下图为本项目的基本架构图: 1.上图中大框架为经典的MVC模式. 2.Co ...
- SPOJ4491. Primes in GCD Table(gcd(a,b)=d素数,(1<=a<=n,1<=b<=m))加强版
SPOJ4491. Primes in GCD Table Problem code: PGCD Johnny has created a table which encodes the result ...
- Objective-C三种定时器CADisplayLink / NSTimer / GCD的使用
OC中的三种定时器:CADisplayLink.NSTimer.GCD 我们先来看看CADiskplayLink, 点进头文件里面看看, 用注释来说明下 @interface CADisplayLin ...
- iOS 多线程之GCD的使用
在iOS开发中,遇到耗时操作,我们经常用到多线程技术.Grand Central Dispatch (GCD)是Apple开发的一个多核编程的解决方法,只需定义想要执行的任务,然后添加到适当的调度队列 ...
随机推荐
- Django错误 OperationalError: no such column: xxx
模型前后操作如下: 第一次迁移: class Snippet(models.Model): created = models.DateTimeField(auto_now_add=True) titl ...
- 通过logger命令记录日志
通过logger命令记录日志 logger是一个shell命令接口,可以通过该接口使用Syslog的系统日志模块,还可以从命令行直接向系统日志文件写入一行信息. ------------------- ...
- 【以前的空间】BZOJ2733[HNOI2012]永无乡
启发式合并?! 似乎当时写并查集的时候就有看到过类似于把小并查集并到大并查集上的说法,原来这就是启发式…… 具体做法就是把小树里面的一个个拿出来,然后加到大树里面去(裸的不敢相信) const max ...
- POJ3281:Dining——题解
http://poj.org/problem?id=3281 题目大意: N牛,F种吃的D种喝的,牛可以在它喜欢的吃的喝的选一组,之后就不能选这个吃的喝的. 问最多满足多少人. ——————————— ...
- C++中的显示类型转换
本文参考了<C++ Primer(中文 第5版)>.<王道程序员求职宝典>以及网上相关博客,结合自己的理解写成.个人水平有限,若有错误欢迎指出. C++中显示转换也成为强制类型 ...
- React的getDefaultProps和getInitialState
getDefaultProps 不是获取默认props,而是设置默认props,主要用在ES5的React写法中getInitialState 不是获取默认State,而是设置初始的state,主要是 ...
- Linux服务器上nginx安装的完整步骤
1.环境准备: 服务器系统版本:CentOS 6.5 nginx软件版本:nginx-1.13.9 2.资源准备: nginx下载地址:http://nginx.org/download/nginx- ...
- matlab特殊符号表
Character Sequence Symbol Character Sequence Symbol Character Sequence Symbol \alpha α \upsilon υ \s ...
- 主机不能访问虚拟机CentOS中的站点
主机能ping通虚拟机 虚拟机也能ping通主机 主机不能telenet通虚拟机 原因:虚拟机开启了防火墙, 解决办法:关闭虚拟机防火墙. Centos 7 firewall 命令: 查看已经开放的端 ...
- Maven命令创建java项目
------------------------------java项目搭建--------------------------- 使用Maven构建一个简单的Java项目 1.进入命令行,执行下面的 ...