GCD是iOS的一种底层多线程机制,今天总结一下GCD的常用API和概念,希望对大家的学习起到帮助作用。 GCD队列的概念 在多线程开发当中,程序员只要将想做的事情定义好,并追加到DispatchQueue(派发队列)当中就好了。 派发队列分为两种,一种是串行队列(SerialDispatchQueue),一种是并行队列(ConcurrentDispatchQueue)。 一个任务就是一个block,比如,将任务添加到队列中的代码是: 1 dispatch_async(queue, block); 当给queue添加多个任务时,如果queue是串行队列,则它们按顺序一个个执行,同时处理的任务只有一个。 当queue是并行队列时,不论第一个任务是否结束,都会立刻开始执行后面的任务,也就是可以同时执行多个任务。 但是并行执行的任务数量取决于XNU内核,是不可控的。比如,如果同时执行10个任务,那么10个任务并不是开启10个线程,线程会根据任务执行情况复用,由系统控制。 获取队列 系统提供了两个队列,一个是MainDispatchQueue,一个是GlobalDispatchQueue。 前者会将任务插入主线程的RunLoop当中去执行,所以显然是个串行队列,我们可以使用它来更新UI。 后者则是一个全局的并行队列,有高、默认、低和后台4个优先级。 它们的获取方式如下: 1 dispatch_queue_t queue = dispatch_get_main_queue(); 2 3 dispatch queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRORITY_DEFAULT, 0) 执行异步任务 1 dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0); 2 dispatch_async(queue, ^{ 3 //... 4 }); 这个代码片段直接在子线程里执行了一个任务块。使用GCD方式任务是立即开始执行的 它不像操作队列那样可以手动启动,同样,缺点也是它的不可控性。 令任务只执行一次 1 + (id)shareInstance { 2 static dispatch_once_t onceToken; 3 dispatch_once(&onceToken, ^{ 4 _shareInstance = [[self alloc] init]; 5 }); 6 } 这种只执行一次且线程安全的方式经常出现在单例构造器当中。 任务组 有时候,我们希望多个任务同时(在多个线程里)执行,再他们都完成之后,再执行其他的任务, 于是可以建立一个分组,让多个任务形成一个组,下面的代码在组中多个任务都执行完毕之后再执行后续的任务: 1 dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0); 2 dispatch_group_t group = dispatch_group_create(); 3 4 dispatch_group_async(group, queue, ^{ NSLog(@"1"); }); 5 dispatch_group_async(group, queue, ^{ NSLog(@"2"); }); 6 dispatch_group_async(group, queue, ^{ NSLog(@"3"); }); 7 dispatch_group_async(group, queue, ^{ NSLog(@"4"); }); 8 dispatch_group_async(group, queue, ^{ NSLog(@"5"); }); 9 10 dispatch_group_notify(group, dispatch_get_main_queue(), ^{ NSLog(@"done"); }); 延迟执行任务 1 dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(10 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{ 2 //... 3 }); 这段代码将会在10秒后将任务插入RunLoop当中。 dispatch_asycn和dispatch_sync 先前已经有过一个使用dispatch_async执行异步任务的一个例子,下面来看一段代码: 1 dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0); 2 3 dispatch_async(queue, ^{ 4 NSLog(@"1"); 5 }); 6 7 NSLog(@"2"); 这段代码首先获取了全局队列,也就是说,dispatch_async当中的任务被丢到了另一个线程里去执行,async在这里的含义是,当当前线程给子线程分配了block当中的任务之后,当前线程会立即执行,并不会发生阻塞,也就是异步的。那么,输出结果不是12就是21,因为我们没法把控两个线程RunLoop里到底是怎么执行的。 类似的,还有一个“同步”方法dispatch_sync,代码如下: 1 dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0); 2 3 dispatch_sync(queue, ^{ 4 NSLog(@"1"); 5 }); 6 7 NSLog(@"2"); 这就意味着,当主线程将任务分给子线程后,主线程会等待子线程执行完毕,再继续执行自身的内容,那么结果显然就是12了。 需要注意的一点是,这里用的是全局队列,那如果把dispatch_sync的队列换成主线程队列会怎么样呢: 1 dispatch_queue_t queue = dispatch_get_main_queue(); 2 dispatch_sync(queue, ^{ 3 NSLog(@"1"); 4 }); 这段代码会发生死锁,因为: 1.主线程通过dispatch_sync把block交给主队列后,会等待block里的任务结束再往下走自身的任务, 2.而队列是先进先出的,block里的任务也在等待主队列当中排在它之前的任务都执行完了再走自己。 这种循环等待就形成了死锁。所以在主线程当中使用dispatch_sync将任务加到主队列是不可取的。 创建队列 我们可以使用系统提供的函数获取主串行队列和全局并行队列,当然也可以自己手动创建串行和并行队列,代码为: 1 dispatch_queue_t mySerialDispatchQueue = dispatch_queue_create("com.Steak.GCD", DISPATCH_QUEUE_SERIAL); 2 dispatch_queue_t myConcurrentDispatchQueue = dispatch_queue_create("com.Steak.GCD", DISPATCH_QUEUE_CONCURRENT); 在MRC下,手动创建的队列是需要释放的 1 dispatch_release(myConcurrentDispatchQueue); 手动创建的队列和默认优先级全局队列优先级等同,如果需要修改队列的优先级,需要: 1 dispatch_queue_t myConcurrentDispatchQueue = dispatch_queue_create("com.Steak.GCD", DISPATCH_QUEUE_CONCURRENT); 2 dispatch_queue_t targetQueue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_BACKGROUND, 0); 3 dispatch_set_target_queue(myConcurrentDispatchQueue, targetQueue); 上面的代码修改队列的优先级为后台级别,即与默认的后台优先级的全局队列等同。 串行、并行队列与读写安全性 在向串行队列(SerialDispatchQueue)当中加入多个block任务后,一次只能同时执行一个block,如果生成了n个串行队列,并且向每个队列当中都添加了任务,那么系统就会启动n个线程来同时执行这些任务。 对于串行队列,正确的使用时机,是在需要解决数据/文件竞争问题时使用它。比如,我们可以令多个任务同时访问一块数据,这样会出现冲突,也可以把每个操作都加入到一个串行队列当中,因为串行队列一次只能执行一个线程的任务,所以不会出现冲突。 但是考虑到串行队列会因为上下文切换而拖慢系统性能,所以我们还是很期望采用并行队列的,来看下面的示例代码: |