一 GCD简介
GCD(Grand Central Dispatch)是苹果为实现并发编程提供的新技术。从基本功能上讲,GCD有点像NSOperationQueue,他们都允许程序将任务切分为多个单一任务然后提交至工作队列来并发地或者串行地执行。但是GCD比之NSOpertionQueue更底层更高效。
GCD的API很大程度上基于block,当然,GCD也可以脱离block来使用,比如使用传统c机制提供函数指针和上下文指针。实践证明,当配合block使用时,GCD非常简单易用且能发挥其最大能力,这也是我们经常在程序中使用的方式。
GCD提供很多超越传统多线程编程的优势:
a.易用: GCD比之thread更加简单易用。由于GCD基于工作单元(work unit)而非像thread那样基于运算,所以GCD可以控制诸如等待任务结束、监视文件描述符、周期执行代码以及工作挂起等任务。作为程序员只要一心关注自己的业务逻辑,可以从繁杂的线程管理的工作中解放出来。基于block的血统导致它能极为简单,可以在不同代码作用域之间传递上下文(闭包特性)。
b.高效率: GCD被实现得如此轻量和优雅,使得它在很多地方比之专门创建消耗资源的线程更实用且快速。这关系到易用性:导致GCD易用的原因有一部分在于你可以不用担心太多的效率问题而仅仅使用它就行了。
c.高性能: GCD自动根据系统负载(比例内存使用情况、CPU个数)来增减线程数量,它甚至可以在一个线程中实现多任务的并发,这就减少了上下文切换带来的开销,增加了计算效率。
注意:1.GCD的层次比线程高,其底层是利用多线程来实现的,苹果将线程交给了系统去管理,这样任务的管理和执行比起线程来更加高效。
2.使用者要做的是定义自己Task任务,然后将其放到合适的分发队列去执行。
3.可以将GCD看作是一种更加轻量级的线程,类似于golang中的协程的概念。GCD的本质是在操作系统层面提供提供并行调度的,这样任务的切换效率要远高于内核的切换效率,它的大部分功能实在运行库里实现的。
二 GCD 分发队列
尽管GCD是纯c语言的,但它被组建成面向对象的风格。GCD对象被称为dispatch object。Dispatch object像Cocoa对象一样是引用计数的。使用dispatch_release和dispatch_retain函数来操作dispatch object的引用计数来进行内存管理。
GCD的基本概念就是dispatch queue。dispatch queue是一个对象,它可以接受任务,并将任务以先到先执行(FIFO)的顺序来执行。dispatch queue可以是并发的或串行的。并发任务会像NSOperationQueue那样基于系统负载来合适地并发进行,串行队列同一时间只执行单一任务。
注意:1. 分发队是FIFO原则来执行任务,但这并不能一定保证任务的最终执行完毕顺序也是按照添加顺序类进行(除非使用的是串行队列),这要看具体的任务内容。
2.多个队列直间是并行执行的。
3.提交到GCD的代码块可以有参数也可以没有,但是返回值必须是void
4。如果任务用C函数进行封装的话 参数和返回值必须都为空 typedef void (*dispatch_function_t)(void *);
3.提交到GCD的代码块可以有参数也可以没有,但是返回值必须是void
4。如果任务用C函数进行封装的话 参数和返回值必须都为空 typedef void (*dispatch_function_t)(void *);
GCD中有三种队列类型:
1.Main Queue 主队列(UI队列): 与主线程功能相同。实际上,提交至main queue的任务会在主线程中执行。main queue可以调用dispatch_get_main_queue()来获得。因为main queue是与主线程相关的,所以这是一个串行队列。
2.Global Queue 全局队列(并发队列):可以同时执行一个或多个任务,而且这些队列是由整个进程共享。进程中默认会自动创建三个全局队列:高、中(默认)、低三个优先级队列。可以调用dispatch_get_global_queue函数传入优先级来访问队列。
3.Serial Qeue 用户自定义队列: 有两种,一种是自定义串行队列,一次只能执行一个任务;一种是自定义同步队列,可以并行执行多个任务。
注意:1.global和main 分发队列由系统创建和管理,不用去关心引用计数,他们属于全局队列由系统去负责。
2.对于自己创建的Serial Qeue需要自己去维护其引用计数(使用ARC就不用了)。
三 任务提交的方式
向GCD中提交任务的方式分两种:
1.同步 dispatch_async
用同步方式提交任务后, 会以阻塞的方式来执行任务,直到任务执行完毕才会顺序往下去执行下面的代码。
用同步方式提交任务后, 会以阻塞的方式来执行任务,直到任务执行完毕才会顺序往下去执行下面的代码。
例如:
A ...................
dispatch_sync(mainQueue, TaskOne);
B .................
当执行完A代码片段后,向主队列提交了一个任务,这样会阻塞程序,直到TaskOne执行完毕后,才会执行B代码片段。
异步 dispatch_async
提交任务后会立即返回,顺序往下执行。
例如:
A ...................
dispatch_async(mainQueue, TaskOne);
B .................
当执行完A代码片段后,向主队列提交了一个任务,这时不管TaskOne是否执行完毕,程序都会立即返回,执行B代码片段。
四 总结
使用GCD可以极大的简化并行编程的方式,将底层的多线程交由系统去管理,这样可以充分利用多核硬件,提高程序的执行效率。下一节将使用代码详细介绍GCD使用。