GCD: 基本概念和Dispatch Queue 【转】

什么是GCD?

Grand Central
Dispatch或者GCD,是一套低层API,提供了一种新的方法来进行并发程序编写。从基本功能上讲,GCD有点像
NSOperationQueue,他们都允许程序将任务切分为多个单一任务然后提交至工作队列来并发地或者串行地执行。GCD比之
NSOpertionQueue更底层更高效,并且它不是Cocoa框架的一部分。

除了代码的平行执行能力,GCD还提供高度集成的事件控制系统。可以设置句柄来响应文件描述符、mach ports(Mach port 用于 OS X上的进程间通讯)、进程、计时器、信号、用户生成事件。这些句柄通过GCD来并发执行。

GCD的API很大程度上基于block,当然,GCD也可以脱离block来使用,比如使用传统c机制提供函数指针和上下文指针。实践证明,当配合block使用时,GCD非常简单易用且能发挥其最大能力。

你可以在Mac上敲命令“man dispatch”来获取GCD的文档。

为何使用?

GCD提供很多超越传统多线程编程的优势:

  1. 易用: GCD比之thread跟简单易用。由于GCD基于work unit而非像thread那样基于运算,所以GCD可以控制诸如等待任务结束、监视文件描述符、周期执行代码以及工作挂起等任务。基于block的血统导致它能极为简单得在不同代码作用域之间传递上下文。
  2. 效率: GCD被实现得如此轻量和优雅,使得它在很多地方比之专门创建消耗资源的线程更实用且快速。这关系到易用性:导致GCD易用的原因有一部分在于你可以不用担心太多的效率问题而仅仅使用它就行了。
  3. 性能: GCD自动根据系统负载来增减线程数量,这就减少了上下文切换以及增加了计算效率。

Dispatch Objects

尽管GCD是纯c语言的,但它被组建成面向对象的风格。GCD对象被称为dispatch object。Dispatch
object像Cocoa对象一样是引用计数的。使用dispatch_release和dispatch_retain函数来操作dispatch
object的引用计数来进行内存管理。但主意不像Cocoa对象,dispatch
object并不参与垃圾回收系统,所以即使开启了GC,你也必须手动管理GCD对象的内存。

Dispatch queues 和 dispatch sources(后面会介绍到)可以被挂起和恢复,可以有一个相关联的任意上下文指针,可以有一个相关联的任务完成触发函数。可以查阅“man dispatch_object”来获取这些功能的更多信息。

Dispatch Queues

GCD的基本概念就是dispatch queue。dispatch
queue是一个对象,它可以接受任务,并将任务以先到先执行的顺序来执行。dispatch
queue可以是并发的或串行的。并发任务会像NSOperationQueue那样基于系统负载来合适地并发进行,串行队列同一时间只执行单一任务。

GCD中有三种队列类型:

  1. The main queue: 与主线程功能相同。实际上,提交至main queue的任务会在主线程中执行。main queue可以调用dispatch_get_main_queue()来获得。因为main queue是与主线程相关的,所以这是一个串行队列。
  2. Global queues: 全局队列是并发队列,并由整个进程共享。进程中存在三个全局队列:高、中(默认)、低三个优先级队列。可以调用dispatch_get_global_queue函数传入优先级来访问队列。
  3. 用户队列: 用户队列 (GCD并不这样称呼这种队列,
    但是没有一个特定的名字来形容这种队列,所以我们称其为用户队列) 是用函数 dispatch_queue_create 创建的队列.
    这些队列是串行的。正因为如此,它们可以用来完成同步机制, 有点像传统线程中的mutex。

创建队列

要使用用户队列,我们首先得创建一个。调用函数dispatch_queue_create就行了。函数的第一个参数是一个标签,这纯是为了
debug。Apple建议我们使用倒置域名来命名队列,比如“com.dreamingwish.subsystem.task”。这些名字会在崩溃日
志中被显示出来,也可以被调试器调用,这在调试中会很有用。第二个参数目前还不支持,传入NULL就行了。

提交 Job

向一个队列提交Job很简单:调用dispatch_async函数,传入一个队列和一个block。队列会在轮到这个block执行时执行这个block的代码。下面的例子是一个在后台执行一个巨长的任务:

  1. dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
  2. [self goDoSomethingLongAndInvolved];
  3. NSLog(@"Done doing something long and involved");
  4. });

dispatch_async 函数会立即返回, block会在后台异步执行。

当然,通常,任务完成时简单地NSLog个消息不是个事儿。在典型的Cocoa程序中,你很有可能希望在任务完成时更新界面,这就意味着需要在主线
程中执行一些代码。你可以简单地完成这个任务——使用嵌套的dispatch,在外层中执行后台任务,在内层中将任务dispatch到main
queue:

  1. dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
  2. [self goDoSomethingLongAndInvolved];
  3. dispatch_async(dispatch_get_main_queue(), ^{
  4. [textField setStringValue:@"Done doing something long and involved"];
  5. });
  6. });

还有一个函数叫dispatch_sync,它干的事儿和dispatch_async相同,但是它会等待block中的代码执行完成并返回。结合

__block类型修饰符,可以用来从执行中的block获取一个值。例如,你可能有一段代码在后台执行,而它需要从界面控制层获取一个值。那么你可以使
用dispatch_sync简单办到:

  1. __block NSString *stringValue;
  2. dispatch_sync(dispatch_get_main_queue(), ^{
  3. // __block variables aren‘t automatically retained
  4. // so we‘d better make sure we have a reference we can keep
  5. stringValue = [[textField stringValue] copy];
  6. });
  7. [stringValue autorelease];
  8. // use stringValue in the background now

我们还可以使用更好的方法来完成这件事——使用更“异步”的风格。不同于取界面层的值时要阻塞后台线程,你可以使用嵌套的block来中止后台线程,然后从主线程中获取值,然后再将后期处理提交至后台线程:

  1. dispatch_queue_t bgQueue = myQueue;
  2. dispatch_async(dispatch_get_main_queue(), ^{
  3. NSString *stringValue = [[[textField stringValue] copy] autorelease];
  4. dispatch_async(bgQueue, ^{
  5. // use stringValue in the background now
  6. });
  7. });

取决于你的需求,myQueue可以是用户队列也可以使全局队列。

不再使用锁(Lock)

用户队列可以用于替代锁来完成同步机制。在传统多线程编程中,你可能有一个对象要被多个线程使用,你需要一个锁来保护这个对象:

  1. NSLock *lock;

访问代码会像这样:

  1. - (id)something
  2. {
  3. id localSomething;
  4. [lock lock];
  5. localSomething = [[something retain] autorelease];
  6. [lock unlock];
  7. return localSomething;
  8. }
  9. - (void)setSomething:(id)newSomething
  10. {
  11. [lock lock];
  12. if(newSomething != something)
  13. {
  14. [something release];
  15. something = [newSomething retain];
  16. [self updateSomethingCaches];
  17. }
  18. [lock unlock];
  19. }

使用GCD,可以使用queue来替代:

  1. dispatch_queue_t queue;

要用于同步机制,queue必须是一个用户队列,而非全局队列,所以使用usingdispatch_queue_create初始化一个。然后可以用dispatch_async 或者 dispatch_sync将共享数据的访问代码封装起来:

  1. - (id)something
  2. {
  3. __block id localSomething;
  4. dispatch_sync(queue, ^{
  5. localSomething = [something retain];
  6. });
  7. return [localSomething autorelease];
  8. }
  9. - (void)setSomething:(id)newSomething
  10. {
  11. dispatch_async(queue, ^{
  12. if(newSomething != something)
  13. {
  14. [something release];
  15. something = [newSomething retain];
  16. [self updateSomethingCaches];
  17. }
  18. });
  19. }

值得注意的是dispatch queue是非常轻量级的,所以你可以大用特用,就像你以前使用lock一样。

现在你可能要问:“这样很好,但是有意思吗?我就是换了点代码办到了同一件事儿。”

实际上,使用GCD途径有几个好处:

  1. 平行计算: 注意在第二个版本的代码中, -setSomething:是怎么使用
    dispatch_async的。调用 -setSomething:会立即返回,然后这一大堆工作会在后台执行。如果
    updateSomethingCaches是一个很费时费力的任务,且调用者将要进行一项处理器高负荷任务,那么这样做会很棒。
  2. 安全: 使用GCD,我们就不可能意外写出具有不成对Lock的代码。在常规Lock代码中,我们很可能在解锁之前让代码返回了。使用GCD,队列通常持续运行,你必将归还控制权。
  3. 控制: 使用GCD我们可以挂起和恢复dispatch
    queue,而这是基于锁的方法所不能实现的。我们还可以将一个用户队列指向另一个dspatch
    queue,使得这个用户队列继承那个dispatch
    queue的属性。使用这种方法,队列的优先级可以被调整——通过将该队列指向一个不同的全局队列,若有必要的话,这个队列甚至可以被用来在主线程上执行
    代码。
  4. 集成: GCD的事件系统与dispatch queue相集成。对象需要使用的任何事件或者计时器都可以从该对象的队列中指向,使得这些句柄可以自动在该队列上执行,从而使得句柄可以与对象自动同步。

总结

现在你已经知道了GCD的基本概念、怎样创建dispatch queue、怎样提交Job至dispatch
queue以及怎样将队列用作线程同步。接下来我会向你展示如何使用GCD来编写平行执行代码来充分利用多核系统的性能^
^。我还会讨论GCD更深层的东西,包括事件系统和queue targeting。

GCD: 基本概念和Dispatch Queue 【转】,布布扣,bubuko.com

时间: 2024-10-10 21:35:08

GCD: 基本概念和Dispatch Queue 【转】的相关文章

GCD基本概念和Dispatch Queue

什么是GCD? Grand Central Dispatch或者GCD,是?一套低层API,提供了?一种新的?方法来进?行并发程序编写.从基本功能上讲,GCD有点像NSOperationQueue,他们都允许程序将 任务切分为多个单?一任务然后提交?至?工作队列来并发地或者串?行地执?行.GCD?比之NSOpertionQueue更底层更?高效,并且它不是Cocoa框架的?一部分. 除了代码的平?行执?行能?力,GCD还提供?高度集成的事件控制系统.可以设置句柄来响应?文件描述符.mach po

IOS开发 GCD介绍: 基本概念和Dispatch Queue

iOS的三种多线程技术 1.NSThread 每个NSThread对象对应一个线程,量级较轻(真正的多线程) 2.以下两点是苹果专门开发的“并发”技术,使得程序员可以不再去关心线程的具体使用问题 ØNSOperation/NSOperationQueue 面向对象的线程技术 ØGCD —— Grand Central Dispatch(派发) 是基于C语言的框架,可以充分利用多核,是苹果推荐使用的多线程技术 以上这三种编程方式从上到下,抽象度层次是从低到高的,抽象度越高的使用越简单,也是Appl

GCD介绍(一):基本概念和Dispatch Queues

什么是GCD? Grand Central Dispatch或者GCD,是一套低层API,提供了一种新的方法来进行并发程序编写.从基本功能上讲,GCD有点像NSOperationQueue,他们都允许程序将任务切分为多个单一任务然后提交至工作队列来并发地或者串行地执行.GCD比之NSOpertionQueue更底层更高效,并且它不是Cocoa框架的一部分. 除了代码的平行执行能力,GCD还提供高度集成的事件控制系统.可以设置句柄来响应文件描述符.mach ports(Mach port 用于 O

GCD之dispatch queue深入浅出

GCD之dispatch queue深入浅出 http://blog.csdn.net/samuelltk/article/details/9452203

ios多线程操作(四)—— GCD核心概念

GCD全称Grand Central Dispatch,可译为"大派发中枢调度器",以纯C语言写成,提供了许多非常强大的函数.GCD是苹果公司为多核的并行运算提出的解决方案,它可以自动利用更多的CPU内核来参与运算,会自动管理线程的生命周(创建线程.调度任务.销毁线程),而程序员只需要告诉GCD想要执行什么任务,不需要编写任何线程管理代码! GCD中有两个核心概念,一是任务,二是队列. 任务:要执行什么样的操作.任务都是预先以Block封装好准备要执行的一段代码. 队列:用来存放任务,

iOS多线程编程(四)------ GCD(Grand Central Dispatch)

一.简介 是基于C语言开发的一套多线程开发机制,也是目前苹果官方推荐的多线程开发方法,用起来也最简单,只是它基于C语言开发,并不像NSOperation是面向对象的开发,而是完全面向过程的.如果使用GCD,完全由系统管理线程,我们不需要编写线程代码.只需定义想要执行的任务,然后添加到适当的调度队列(dispatch_queue).GCD会负责创建线程和调度你的任务,系统会直接提供线程管理. 二.任务和队列 GCD中有两个核心概念 (1)任务:执行什么操作 (2)队列:用来存放任务 GCD的使用就

GCD(Grand Central Dispatch)的详解

GCD(Grand Central Dispatch) FIFO(先入先出) Dispatch(派遣,急件,分派) Serial (连续的,连载的,分期偿还的) Dispatch Queue(调度队列) Serial Dispatch Queue(串行调度队列) Concurrent Dispatch Queue(并发调度队列) Main Dispatch Queue(主要调度队列) Global Dispatch Queue(全局调度队列) dispatch_apply是(同步函数) GCD是

装个蒜。学习下dispatch queue

dispatch queue的真髓:能串行,能并行,能同步,能异步以及共享同一个线程池. 接口: GCD是基于C语言的APT.虽然最新的系统版本中GCD对象已经转成了Objective-C对象,但API仍保持纯C接口(加了block扩展).这对实现底层接口是好事,GCD提供了出色而简单的接口. Objective-C类名称为MADispatchQueue,包含四个调用方法: 1. 获取全局共享队列的方法.GCD有多个不同优先级的全局队列,出于简单考虑,我们在实现中保留一个. 2. 串行和并行队列

多线程之GCD(Grand Central Dispatch)

GCD是纯C语言,但又溶有面向对象思想.基于Block. 1.GCD优点: 易用:GCD比thread更简单易用,基于Block的特性导致它能极为简单的在不同代码作用域之间传递上下文. 效率:GCD在很多地方比之专门创建消耗资源的线程更实用.快速. 性能:GCD自动根据系统负载来增减线程数量,减少了上下文切换以及提高了计算效率. 2.GCD的使用 2.1.dispatch async异步操作 2.1.1.定义想要执行的操作(任务),追加到适当的队列中(Dispatch Queue) 2.12.Q