`
啸笑天
  • 浏览: 3467946 次
  • 性别: Icon_minigender_1
  • 来自: China
社区版块
存档分类
最新评论

GCD使用经验与技巧浅谈

 
阅读更多

前言

GCD(Grand Central Dispatch)可以说是Mac、iOS开发中的一大“利器”,本文就总结一些有关使用GCD的经验与技巧。

dispatch_once_t必须是全局或static变量

这一条算是“老生常谈”了,但我认为还是有必要强调一次,毕竟非全局或非static的dispatch_once_t变量在使用时会导致非常不好排查的bug,正确的如下:

1
2
3
4
5
//静态变量,保证只有一份实例,才能确保只执行一次
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
   //单例代码 
});

其实就是保证dispatch_once_t只有一份实例。

dispatch_queue_create的第二个参数

dispatch_queue_create,创建队列用的,它的参数只有两个,原型如下:

1
dispatch_queue_t dispatch_queue_create ( const char *label, dispatch_queue_attr_t attr );

在网上的大部分教程里(甚至Apple自己的文档里),都是这么创建串行队列的:

1
dispatch_queue_t queue = dispatch_queue_create("com.example.MyQueue", NULL);

看,第二个参数传的是“NULL”。 但是dispatch_queue_attr_t类型是有已经定义好的常量的,所以我认为,为了更加的清晰、严谨,最好如下创建队列:

1
2
3
4
//串行队列
dispatch_queue_t queue = dispatch_queue_create("com.example.MyQueue", DISPATCH_QUEUE_SERIAL);
//并行队列
dispatch_queue_t queue = dispatch_queue_create("com.example.MyQueue", DISPATCH_QUEUE_CONCURRENT);

常量就是为了使代码更加“易懂”,更加清晰,既然有,为啥不用呢~

dispatch_after是延迟提交,不是延迟运行

先看看官方文档的说明:

1
Enqueue a block for execution at the specified time.

Enqueue,就是入队,指的就是将一个Block在特定的延时以后,加入到指定的队列中,不是在特定的时间后立即运行!。

看看如下代码示例:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
//创建串行队列
dispatch_queue_t queue = dispatch_queue_create("me.tutuge.test.gcd"DISPATCH_QUEUE_SERIAL);
//立即打印一条信息        
NSLog(@"Begin add block...");        
//提交一个block
dispatch_async(queue, ^{
    //Sleep 10秒
    [NSThread sleepForTimeInterval:10];
    NSLog(@"First block done...");
});        
//5 秒以后提交block
dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(5 * NSEC_PER_SEC)), queue, ^{
    NSLog(@"After...");
});

结果如下:

1
2
3
2015-03-31 20:57:27.122 GCDTest[45633:1812016] Begin add block...
2015-03-31 20:57:37.127 GCDTest[45633:1812041] First block done...
2015-03-31 20:57:37.127 GCDTest[45633:1812041] After...

从结果也验证了,dispatch_after只是延时提交block,并不是延时后立即执行。所以想用dispatch_after精确控制运行状态的朋友可要注意了~

正确创建dispatch_time_t

用dispatch_after的时候就会用到dispatch_time_t变量,但是如何创建合适的时间呢?答案就是用dispatch_time函数,其原型如下:

1
dispatch_time_t dispatch_time ( dispatch_time_t when, int64_t delta );

第一个参数一般是DISPATCH_TIME_NOW,表示从现在开始。

那么第二个参数就是真正的延时的具体时间。

这里要特别注意的是,delta参数是“纳秒!”,就是说,延时1秒的话,delta应该是“1000000000”=。=,太长了,所以理所当然系统提供了常量,如下:

1
2
3
#define NSEC_PER_SEC 1000000000ull
#define USEC_PER_SEC 1000000ull
#define NSEC_PER_USEC 1000ull

关键词解释:

  • NSEC:纳秒。

  • USEC:微妙。

  • SEC:秒

  • PER:每

所以:

  1. NSEC_PER_SEC,每秒有多少纳秒。

  2. USEC_PER_SEC,每秒有多少毫秒。(注意是指在纳秒的基础上)

  3. NSEC_PER_USEC,每毫秒有多少纳秒。

所以,延时1秒可以写成如下几种:

dispatch_time(DISPATCH_TIME_NOW, 1 * NSEC_PER_SEC);

dispatch_time(DISPATCH_TIME_NOW, 1000 * USEC_PER_SEC);

dispatch_time(DISPATCH_TIME_NOW, USEC_PER_SEC * NSEC_PER_USEC);

最后一个“USEC_PER_SEC * NSEC_PER_USEC”,翻译过来就是“每秒的毫秒数乘以每毫秒的纳秒数”,也就是“每秒的纳秒数”,所以,延时500毫秒之类的,也就不难了吧~

dispatch_suspend != 立即停止队列的运行

dispatch_suspend,dispatch_resume提供了“挂起、恢复”队列的功能,简单来说,就是可以暂停、恢复队列上的任务。但是这里的“挂起”,并不能保证可以立即停止队列上正在运行的block,看如下例子:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
dispatch_queue_t queue = dispatch_queue_create("me.tutuge.test.gcd", DISPATCH_QUEUE_SERIAL);
//提交第一个block,延时5秒打印。
dispatch_async(queue, ^{
    [NSThread sleepForTimeInterval:5];
    NSLog(@"After 5 seconds...");
});
//提交第二个block,也是延时5秒打印
dispatch_async(queue, ^{
    [NSThread sleepForTimeInterval:5];
    NSLog(@"After 5 seconds again...");
});
//延时一秒
NSLog(@"sleep 1 second...");
[NSThread sleepForTimeInterval:1];
//挂起队列                        
NSLog(@"suspend...");
dispatch_suspend(queue);
//延时10秒                
NSLog(@"sleep 10 second...");
[NSThread sleepForTimeInterval:10];
//恢复队列            
NSLog(@"resume...");
dispatch_resume(queue);

运行结果如下:

1
2
3
4
5
6
2015-04-01 00:32:09.903 GCDTest[47201:1883834] sleep 1 second...
2015-04-01 00:32:10.910 GCDTest[47201:1883834] suspend...
2015-04-01 00:32:10.910 GCDTest[47201:1883834] sleep 10 second...
2015-04-01 00:32:14.908 GCDTest[47201:1883856] After 5 seconds...
2015-04-01 00:32:20.911 GCDTest[47201:1883834] resume...
2015-04-01 00:32:25.912 GCDTest[47201:1883856] After 5 seconds again...

可知,在dispatch_suspend挂起队列后,第一个block还是在运行,并且正常输出。

结合文档,我们可以得知,dispatch_suspend并不会立即暂停正在运行的block,而是在当前block执行完成后,暂停后续的block执行。

所以下次想暂停正在队列上运行的block时,还是不要用dispatch_suspend了吧~

“同步”的dispatch_apply

dispatch_apply的作用是在一个队列(串行或并行)上“运行”多次block,其实就是简化了用循环去向队列依次添加block任务。但是我个人觉得这个函数就是个“坑”,先看看如下代码运行结果:

1
2
3
4
5
6
7
8
//创建异步串行队列
dispatch_queue_t queue = dispatch_queue_create("me.tutuge.test.gcd", DISPATCH_QUEUE_SERIAL);
//运行block3次
dispatch_apply(3, queue, ^(size_t i) {
    NSLog(@"apply loop: %zu", i);
});
//打印信息
NSLog(@"After apply");

运行的结果是:

1
2
3
4
2015-04-01 00:55:40.854 GCDTest[47402:1893289] apply loop: 0
2015-04-01 00:55:40.856 GCDTest[47402:1893289] apply loop: 1
2015-04-01 00:55:40.856 GCDTest[47402:1893289] apply loop: 2
2015-04-01 00:55:40.856 GCDTest[47402:1893289] After apply

看,明明是提交到异步的队列去运行,但是“After apply”居然在apply后打印,也就是说,dispatch_apply将外面的线程(main线程)“阻塞”了!

查看官方文档,dispatch_apply确实会“等待”其所有的循环运行完毕才往下执行=。=,看来要小心使用了。

避免死锁!

dispatch_sync导致的死锁

涉及到多线程的时候,不可避免的就会有“死锁”这个问题,在使用GCD时,往往一不小心,就可能造成死锁,看看下面的“死锁”例子:

1
2
3
4
//在main线程使用“同步”方法提交Block,必定会死锁。
dispatch_sync(dispatch_get_main_queue(), ^{
    NSLog(@"I am block...");
});

你可能会说,这么低级的错误,我怎么会犯,那么,看看下面的:

1
2
3
4
5
6
7
8
9
10
11
12
13
- (void)updateUI1 {
    dispatch_sync(dispatch_get_main_queue(), ^{
        NSLog(@"Update ui 1");
         
        //死锁!
        [self updateUI2];
    });
}
- (void)updateUI2 {
    dispatch_sync(dispatch_get_main_queue(), ^{
        NSLog(@"Update ui 2");
    });
}

在你不注意的时候,嵌套调用可能就会造成死锁!所以为了“世界和平”=。=,我们还是少用dispatch_sync吧。

dispatch_apply导致的死锁!

啥,dispatch_apply导致的死锁?。。。是的,前一节讲到,dispatch_apply会等循环执行完成,这不就差不多是阻塞了吗。看如下例子:

1
2
3
4
5
6
7
8
9
10
dispatch_queue_t queue = dispatch_queue_create("me.tutuge.test.gcd", DISPATCH_QUEUE_SERIAL);
        
dispatch_apply(3, queue, ^(size_t i) {
NSLog(@"apply loop: %zu", i);
    
    //再来一个dispatch_apply!死锁!      
dispatch_apply(3, queue, ^(size_t j) {
NSLog(@"apply loop inside %zu", j);
});
});

这端代码只会输出“apply loop: 1”。。。就没有然后了=。=

所以,一定要避免dispatch_apply的嵌套调用。

灵活使用dispatch_group

很多时候我们需要等待一系列任务(block)执行完成,然后再做一些收尾的工作。如果是有序的任务,可以分步骤完成的,直接使用串行队列就行。但是如果是一系列并行执行的任务呢?这个时候,就需要dispatch_group帮忙了~总的来说,dispatch_group的使用分如下几步:

  1. 创建dispatch_group_t

  2. 添加任务(block)

  3. 添加结束任务(如清理操作、通知UI等)

下面着重讲讲在后面两步。

添加任务

添加任务可以分为以下两种情况:

自己创建队列:使用dispatch_group_async

无法直接使用队列变量(如使用AFNetworking添加异步任务):使用dispatch_group_enter,dispatch_group_leave

自己创建队列时,当然就用dispatch_group_async函数,简单有效,简单例子如下:

1
2
3
4
//省去创建group、queue代码。。。
dispatch_group_async(group, queue, ^{
    //Do you work...
});

当你无法直接使用队列变量时,就无法使用dispatch_group_async了,下面以使用AFNetworking时的情况:

1
2
3
4
5
6
7
8
9
10
11
12
13
AFHTTPRequestOperationManager *manager = [AFHTTPRequestOperationManager manager];
//Enter group
dispatch_group_enter(group);
[manager GET:@"http://www.baidu.com" parameters:nil success:^(AFHTTPRequestOperation *operation, id responseObject) {
    //Deal with result...
    //Leave group
    dispatch_group_leave(group);
}    failure:^(AFHTTPRequestOperation *operation, NSError *error) {
    //Deal with error...
    //Leave group
    dispatch_group_leave(group);
}];
//More request...

使用dispatch_group_enter,dispatch_group_leave就可以方便的将一系列网络请求“打包”起来~

添加结束任务

添加结束任务也可以分为两种情况,如下:

  1. 在当前线程阻塞的同步等待:dispatch_group_wait。

  2. 添加一个异步执行的任务作为结束任务:dispatch_group_notify

这两个比较简单,就不再贴代码了=。=

使用dispatch_barrier_async,dispatch_barrier_sync的注意事项

dispatch_barrier_async的作用就是向某个队列插入一个block,当目前正在执行的block运行完成后,阻塞这个block后面添加的block,只运行这个block直到完成,然后再继续后续的任务,有点“唯我独尊”的感觉=。=

值得注意的是:

dispatchbarrier\(a)sync只在自己创建的并发队列上有效,在全局(Global)并发队列、串行队列上,效果跟dispatch_(a)sync效果一样。

既然在串行队列上跟dispatch_(a)sync效果一样,那就要小心别死锁!

dispatch_set_context与dispatch_set_finalizer_f的配合使用

dispatch_set_context可以为队列添加上下文数据,但是因为GCD是C语言接口形式的,所以其context参数类型是“void *”。也就是说,我们创建context时有如下几种选择:

用C语言的malloc创建context数据。

用C++的new创建类对象。

用Objective-C的对象,但是要用__bridge等关键字转为Core Foundation对象。

以上所有创建context的方法都有一个必须的要求,就是都要释放内存!,无论是用free、delete还是CF的CFRelease,我们都要确保在队列不用的时候,释放context的内存,否则就会造成内存泄露。

所以,使用dispatch_set_context的时候,最好结合dispatch_set_finalizer_f使用,为队列设置“析构函数”,在这个函数里面释放内存,大致如下:

1
2
3
4
5
6
7
8
9
void cleanStaff(void *context) {
    //释放context的内存!
    //CFRelease(context);
    //free(context);
    //delete context;
}
...
//在队列创建后,设置其“析构函数”
dispatch_set_finalizer_f(queue, cleanStaff);

详细用法,请看我之前写的Blog为GCD队列绑定NSObject类型上下文数据-利用__bridge_retained(transfer)转移内存管理权

总结

其实本文更像是总结了GCD中的“坑”=。=

至于经验,总结一条,就是使用任何技术,都要研究透彻,否则后患无穷啊~

参考

 

 

感谢:http://tutuge.me/2015/04/03/something-about-gcd/

分享到:
评论
4 楼 啸笑天 2015-08-18  
Grand Central Dispatch 基础教程:Part 1/2  http://www.jianshu.com/p/50c060bab0ff

Grand Central Dispatch 基础教程:Part 2/2  http://www.jianshu.com/p/6185d3753dd8
3 楼 啸笑天 2015-08-17  
http://www.jianshu.com/p/0b0d9b1f1f19 关于iOS多线程,你看我就够了
2 楼 啸笑天 2015-08-06  
对于dispatch_queue_t 对象来说,我们应该这么写
#if OS_OBJECT_USE_OBJC
@property (strong, nonatomic) dispatch_queue_t barrierQueue;
#else
@property (assign, nonatomic) dispatch_queue_t barrierQueue;
#endif

在dealloc方法中应该加上
#if !OS_OBJECT_USE_OBJC   //这个宏是在sdk6.0之后才有的,如果是之前的,则OS_OBJECT_USE_OBJC为0
    dispatch_release(_barrierQueue);
#endif
当然了,也可以使用
#if __IPHONE_OS_VERSION_MIN_REQUIRED < 60000 // 6.0sdk之前
    dispatch_release(_barrierQueue);
#endif
这里的宏  __IPHONE_OS_VERSION_MIN_REQUIRED  就是我们在工程的设置项里设置的最低部署sdk版本


原因就是  对于最低sdk版本>=ios6.0来说,GCD对象已经纳入了ARC的管理范围,我们就不需要再手工调用 dispatch_release了,否则的话,在sdk<6.0的时候,即使我们开启了ARC,这个宏OS_OBJECT_USE_OBJC 也是没有的,也就是说这个时候,GCD对象还必须得自己管理

如果你部署的最低目标低于 iOS 6.0 or Mac OS X 10.8
你应该自己管理GCD对象,使用(dispatch_retain,dispatch_release),ARC并不会去管理它们

如果你部署的最低目标是 iOS 6.0 or Mac OS X 10.8 或者更高的
ARC已经能够管理GCD对象了,这时候,GCD对象就如同普通的OC对象一样,不应该使用dispatch_retain ordispatch_release 
1 楼 啸笑天 2015-08-06  
func dispatch_barrier_async(_ queue: dispatch_queue_t, _ block: dispatch_block_t):
这个方法重点是你传入的 queue,当你传入的 queue 是通过 DISPATCH_QUEUE_CONCURRENT 参数自己创建的 queue 时,这个方法会阻塞这个 queue(注意是阻塞 queue ,而不是阻塞当前线程),一直等到这个 queue 中排在它前面的任务都执行完成后才会开始执行自己,自己执行完毕后,再会取消阻塞,使这个 queue 中排在它后面的任务继续执行。
如果你传入的是其他的 queue, 那么它就和 dispatch_async 一样了。
func dispatch_barrier_sync(_ queue: dispatch_queue_t, _ block: dispatch_block_t):
这个方法的使用和上一个一样,传入 自定义的并发队列(DISPATCH_QUEUE_CONCURRENT),它和上一个方法一样的阻塞 queue,不同的是 这个方法还会 阻塞当前线程。
如果你传入的是其他的 queue, 那么它就和 dispatch_sync 一样了。

相关推荐

    GCD使用经验与技巧浅谈 - CocoaChina_让移动开发更简单1

    本文将深入探讨GCD的一些使用经验和技巧。 首先,GCD的核心概念是队列(queue)。队列分为两种类型:串行队列(serial queue)和并行队列(concurrent queue)。串行队列会按照任务添加的顺序逐一执行,而并行队列...

    swift-GCD使用大全

    10. **GCD与CancellationToken** Swift 5.5引入了CancellationToken,与GCD结合,可以更优雅地取消异步任务。 11. **GCD实战示例** 通过`GCD_Demo-master`中的项目,你可以看到如何在实际应用中使用上述知识点,...

    GCD使用的相关方法

    6. **GCD与NSOperationQueue** NSOperationQueue是Objective-C中类似GCD的机制,提供了更多的高级特性,如任务依赖、取消任务、限制并行度等。两者可以根据项目需求灵活选择。 总结,GCD是苹果平台强大的并发工具...

    大恒GCD-0401M电动台控制器 用户使用手册 Ver1.0(中文版)

    大恒GCD-0401M电动台控制器用户使用手册Ver1.0(中文版) 本资源是大恒GCD-0401M电动台控制器的用户使用手册,版本号为Ver1.0,语言为中文。该手册详细介绍了电动台控制器的组成、主要参数、使用方法、通讯协议、...

    Block及GCD使用

    本分享文档中包含block的基本所有使用实例,及GCD使用

    计算GCD的简单java程序

    在这个改进的版本中,我们首先将数组的第一个元素设为初始GCD,然后遍历数组中的其他元素,依次与当前的GCD进行计算,最终得到整个数组的GCD。 关于标签中的“bootstrap-daterangepicker-master”,这似乎与GCD计算...

    GCD异步获取图片

    2. 任务与队列:GCD中有两种主要概念——任务(block)和队列(queue)。任务是你想要执行的工作,而队列则是存放任务的地方。队列分为串行队列和并行队列。 二、异步获取图片的重要性 1. UI更新:异步加载图片可以...

    iOS GCD多核编程

    在GCD中,可以使用`dispatch_async`函数异步执行任务,这样不会阻塞当前线程。例如,我们可以在后台下载数据,同时主线程继续处理UI: ```swift let queue = DispatchQueue.global(qos: .background) queue.async {...

    GCD 源码分析.pdf

    __builtin_expect() 函数可以与 if 语句结合使用,以提高代码的执行速度。 fastpath 和 slowpath 是两个宏定义,fastpath 用于指定快速路径,slowpath 用于指定慢速路径。fastpath 和 slowpath 可以与 __builtin_...

    GCD基本概念

    1. **GCD与NSOperationQueue的对比:** GCD和NSOperationQueue都可以将任务分解并排队,从而并发或串行执行。不过,GCD更为底层,执行效率更高,且不依赖于Cocoa框架。GCD的API主要基于block,提供了一种简单易用的...

    GCD总结代码

    在实际开发中,GCD的使用往往结合其他技术,如NSOperation和NSOperationQueue,它们提供了更高级别的抽象,可以更好地控制任务依赖关系和优先级。GCD是iOS和macOS开发中的强大工具,合理运用能极大地提升应用的性能...

    ios gcd

    - GCD与Block的结合使得异步编程更加简洁,无需创建单独的类或方法。 - Block可以捕获并保留其作用域内的变量,需要注意循环引用问题。 9. **GCD与OperationQueue对比**: - 虽然GCD提供基础的并发功能,但...

    gcd方法代码测试

    gcd的基础方法使用,线程和队列的使用 gcd的基础方法使用,线程和队列的使用

    gcd_lcm.rar_gcd_gcd l

    标题中的"gcd_lcm.rar_gcd_gcd l"暗示了这个压缩包可能包含与最大公约数(Greatest Common Divisor, GCD)和最小公倍数(Least Common Multiple, LCM)相关的代码或项目,主要关注加法和减法运算。标签"gcd"和"gcd_l...

    gcdtest1示例源码

    GCD是一种高效的任务调度框架,它基于C语言,但被设计为与Objective-C和Swift等Apple的高级编程语言无缝集成。GCD通过提供一种简单的方式来管理和调度并发任务,从而简化了多核处理器和多线程编程。在“gcdtest1示例...

    iOS GCD timer 计时器

    本篇文章将深入探讨iOS GCD timer的相关知识点,并介绍如何封装和使用。 1. **GCD基础** - GCD是Apple提供的底层任务调度框架,基于C语言实现,支持多核处理器的并行处理。 - GCD自动管理线程的创建和销毁,...

    GCD 总结-队列和任务的理解

    GCD 是基于 C 语言的,但在 Objective-C 和 Swift 中都可以使用。它是Apple的系统级并发技术,用于管理线程和调度任务。通过GCD,开发者可以将关注点集中在任务本身,而不是线程管理上,从而提高代码的可读性和可...

    gcd.rar_ Verilog gcd_Verilog GCD_gcd_gcd verilog code_gcd_testbe

    在这个“gcd.rar”压缩包中,包含了一个使用Verilog实现的最大公约数(Greatest Common Divisor, GCD)算法的源代码,以及可能的测试平台或测试用例。 GCD是数学中的一个基本概念,它是指能够同时整除两个或多个非...

    GCD.rar_gcd

    "rar"是常用的压缩文件格式,而"_gcd"可能是强调这个压缩包的内容与GCD计算相关。 描述中提到的"各个源描述的编译顺序:gcd.vhd, gcd_stim.vhd",表明压缩包中包含两个VHDL源文件。"gcd.vhd"很可能是实现GCD算法的...

    iOS GCD详解

    iOS GCD详解 一、什么是GCD? GCD(Grand Central ...使用GCD可以提供很多超越传统多线程编程的优势,如易用性、效率、性能等。GCD自动根据系统负载来增减线程数量,这就减少了上下午切换以及增加了计算效率。

Global site tag (gtag.js) - Google Analytics