关闭

iOS系统GCD学习(3):dispatch_group

411人阅读 评论(0) 收藏 举报
分类:

设想下面的循环:

  1. for(id obj in array)  
  2.     [self doSomethingIntensiveWith:obj];  

  假定 -doSomethingIntensiveWith: 是线程安全的且可以同时执行多个.一个array通常包含多个元素,这样的话,我们可以很简单地使用GCD来平行运算:

  1. dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);  
  2. for(id obj in array)  
  3.     dispatch_async(queue, ^{  
  4.         [self doSomethingIntensiveWith:obj];  
  5.     });  

  如此简单,我们已经在多核心上运行这段代码了。

  当然这段代码并不完美。有时候我们有一段代码要像这样操作一个数组,但是在操作完成后,我们还需要对操作结果进行其他操作:

  1. for(id obj in array)  
  2.      [self doSomethingIntensiveWith:obj];  
  3.  [self doSomethingWith:array];  

  这时候使用GCD的 dispatch_async 就悲剧了.我们还不能简单地使用dispatch_sync来解决这个问题, 因为这将导致每个迭代器阻塞,就完全破坏了平行计算。

  解决这个问题的一种方法是使用dispatch group。一个dispatch group可以用来将多个block组成一组以监测这些Block全部完成或者等待全部完成时发出的消息。使用函数dispatch_group_create来创建,然后使用函数dispatch_group_async来将block提交至一个dispatch queue,同时将它们添加至一个组。所以我们现在可以重新代码:

  1. dispatch_queue_t queue = dispatch_get_global_qeueue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);  
  2. dispatch_group_t group = dispatch_group_create();  
  3. for(id obj in array)  
  4.     dispatch_group_async(group, queue, ^{  
  5.         [self doSomethingIntensiveWith:obj];  
  6.     });  
  7. dispatch_group_wait(group, DISPATCH_TIME_FOREVER);  
  8. dispatch_release(group);  


  [self doSomethingWith:array];如果这些工作可以异步执行,那么我们可以更风骚一点,将函数-doSomethingWith:放在后台执行。我们使用dispatch_group_async函数建立一个block在组完成后执行:


  1. dispatch_queue_t queue = dispatch_get_global_qeueue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);  
  2.   dispatch_group_t group = dispatch_group_create();  
  3.   for(id obj in array)  
  4.       dispatch_group_async(group, queue, ^{  
  5.           [self doSomethingIntensiveWith:obj];  
  6.       });  
  7.   //等group里的task都执行完后执行notify方法里的内容,相当于把wait方法及之后要执行的代码合到一起了  
  8.   dispatch_group_notify(group, queue, ^{  
  9.       [self doSomethingWith:array];  
  10.   });  


  dispatch_release(group);不仅所有数组元素都会被平行操作,后续的操作也会异步执行,并且这些异步运算都会将程序的其他部分考虑在内。注意如果-doSomethingWith:需要在主线程中执行,比如操作GUI,那么我们只要将main queue而非全局队列传给dispatch_group_notify函数就行了。

  还有一段演示dispatch_group的代码:

  1. dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);   
  2. dispatch_group_t group = dispatch_group_create();   
  3. dispatch_group_async(group, queue, ^{   
  4.     [NSThread sleepForTimeInterval:1];   
  5.     NSLog(@"group1");   
  6. });   
  7. dispatch_group_async(group, queue, ^{   
  8.     [NSThread sleepForTimeInterval:2];   
  9.     NSLog(@"group2");   
  10. });   
  11. dispatch_group_async(group, queue, ^{   
  12.     [NSThread sleepForTimeInterval:3];   
  13.     NSLog(@"group3");   
  14. });   
  15. dispatch_group_notify(group, dispatch_get_main_queue(), ^{   
  16.     NSLog(@"updateUi");   
  17. });  
0
0

查看评论
* 以上用户言论只代表其个人观点,不代表CSDN网站的观点或立场
    个人资料
    • 访问:192550次
    • 积分:3282
    • 等级:
    • 排名:第10767名
    • 原创:50篇
    • 转载:413篇
    • 译文:0篇
    • 评论:2条