GCD
Grand Central Dispatch
的简称,由系统管理线程的生命周期-
核心概念
- 任务
- 同步函数(不具备开启线程的能力)
dispatch_sync(dispatch_queue_t queue, dispatch_block_t block);
- 特点:立刻马上执行,如果当前的代码没有执行完毕,则无法执行后面的代码
- 同步函数和主队列会发生死锁
- 异步函数(具备开启线程的能力)
dispatch_async(dispatch_queue_t queue, dispatch_block_t block);
- 特点:当前的代码没有执行完毕,可以继续执行后面的代码
- 同步函数(不具备开启线程的能力)
- 队列
- 并发队列 Concurrent Dispatch Queue
- 使用
dispatch_queue_create
手动创建并发队列 - 使用
dispatch_get_global_queue
函数获得全局的并发队列,无需手动创建
- 使用
- 串行队列 Serial Dispatch Queue
- 使用
dispatch_queue_create
手动创建并发队列 - 使用dispatch_get_main_queue()获得主队列
- 使用
- 并发队列 Concurrent Dispatch Queue
- 任务
-
使用步骤
- 制定任务
- 将任务添加到队列中
-
各种队列的执行效果
注意
使用同步函数往当前串行队列中添加任务,会卡住当前的串行队列 -
线程间通信
- 从子线程回到主线程
- 示例
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{ // 执行耗时的异步操作... dispatch_async(dispatch_get_main_queue(), ^{ // 回到主线程,执行UI刷新操作 }); });
-
GCD常用函数
- 栅栏函数
dispatch_barrier_async(dispatch_queue_t queue, dispatch_block_t block);
- 栅栏函数的作用:控制并发队列中任务的执行顺序
- 栅栏函数中的队列不能是全局的并发队列(苹果官方说的,没有给出原因)
- 延时执行
dispatch_after(dispatch_time(DISPATCH_TIME_NOW,(int64_t)(2.0 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{ // 2秒后执行这里的代码... });
- 一次性代码
- 使用dispatch_once函数能保证某段代码在程序运行过程中只被执行1次
- 一次性代码不能放在懒加载中
- 创建两个对象时,当创建第一个对象,会进入到懒加载中的一次性代码中,但当创建第二个对象时,便不会进入到懒加载的一次性代码中,会导致第二个对象为空
- 语法
static dispatch_once_t onceToken; dispatch_once(&onceToken, ^{ // 只执行1次的代码(这里面默认是线程安全的) });
- 快速迭代
- 使用dispatch_apply函数能进行快速迭代遍历
- 语法
dispatch_apply(10,dispatch_get_global_queue(0,0),^(size_t index){ // 执行10次代码,index顺序不确定 });
- 示例:把一个文件夹的所有内容剪切到另外一个文件夹中
-(void)moveFileWithApply{ // 1. 获得源文件夹的路径 NSString *from = @"/Users/joyce/Desktop/截图"; // 2. 获得目标文件夹的路径 NSString *to = @"/Users/joyce/Desktop/小码哥"; // 3. 通过文件管理者确定文件夹中所有文件的名称 // 3.1 创建文件管理者单例 NSFileManager *manager = [NSFileManager defaultManager]; // 3.2 获取所有文件的名称 NSArray *subpaths = [manager subpathsOfDirectoryAtPath:from error:nil]; NSLog(@"%@", subpaths); // 4. 创建队列 dispatch_queue_t queue = dispatch_get_global_queue(0, 0); // 5. 快速迭代剪切源文件夹中的所有内容 // 5.1 获取剪切内容的个数 NSInteger count = subpaths.count; dispatch_apply(count, queue, ^(size_t index) { // 5.2 获取源文件夹中每个文件的名称 NSString *subpath = subpaths[index]; NSLog(@"%@", subpath); // 5.3 拼接文件的全路径 NSString *fromFullFilePath = [from stringByAppendingPathComponent:subpath]; NSLog(@"%@", fromFullFilePath); // 5.4 剪切到目标文件的全路径 NSString *toFullFilePath = [to stringByAppendingPathComponent:subpath]; // 5.5 执行剪切 [manager moveItemAtPath:fromFullFilePath toPath:toFullFilePath error:nil]; });
- 栅栏函数
}
* GCD队列组
* 同栅栏函数
* 示例:下载两张图片并合成
* 思路
* 下载图片1
* 下载图片2
* 合成图片
* 具体实现步骤
* 创建队列组
* 创建队列(获取全局并发队列)
* 异步函数下载图片1
* 异步函数下载图片2
* 拦截通知,合成图片,依赖前面两步(直接传递主队列参数)
* 代码实现
```objc
-(void)combie{
// 1. 创建队列组
dispatch_group_t group = dispatch_group_create();
// 2. 创建队列
dispatch_queue_t queue = dispatch_get_global_queue(0, 0);
// 3. 异步函数下载图片1
dispatch_group_async(group, queue, ^{
// 确定url
NSURL *url = [NSURL URLWithString:@"http://pic28.nipic.com/20130402/9252150_190139450381_2.jpg"];
// 下载图片的二进制数据到本地
NSData *data = [NSData dataWithContentsOfURL:url];
// 把二进制数据转化成图片
self.image1 = [UIImage imageWithData:data];
});
// 4. 异步函数下载图片2
dispatch_group_async(group, queue, ^{
// 确定url
NSURL *url = [NSURL URLWithString:@"http://pic14.nipic.com/20110522/7411759_164157418126_2.jpg"];
// 下载图片的二进制数据到本地
NSData *data = [NSData dataWithContentsOfURL:url];
// 把二进制数据转化成图片
self.image2 = [UIImage imageWithData:data];
});
// 5. 拦截通知,合成图片,依赖3&4
dispatch_group_notify(group, dispatch_get_main_queue(), ^{
// 开启图形上下文
UIGraphicsBeginImageContext(CGSizeMake(240, 240));
// 画图1
[self.image1 drawInRect:CGRectMake(0, 0, 240, 120)];
// 画图2
[self.image2 drawInRect:CGRectMake(0, 120, 240, 120)];
// 生成图片
UIImage *image = UIGraphicsGetImageFromCurrentImageContext();
// 关闭图形上下文
UIGraphicsEndImageContext();
// 设置imageViewd的image属性
self.imageView.image = image;
});
}
-
dispatch_group_notify
底层是dispatch_group_enter``和
dispatch_group_leave` 配对使用