iOS多线程 - GCD_Group
在我们的实际项目当中,经常会有这样的需求:在多个任务异步处理之后,我们需要统一的一个回调通知,来告诉我所有的任务都已经结束了。
这个时候我们就可以用到dispatch_group
。它有个作用就是,当所有任务完成之后会调用“
NSLog(@"执行GCD");
// 创建一个并行的queue
dispatch_queue_t queue = dispatch_queue_create("gcd.group", DISPATCH_QUEUE_CONCURRENT);
// 创建group
dispatch_group_t group = dispatch_group_create();
dispatch_group_async(group, queue, ^{
NSLog(@"star任务1");
[NSThread sleepForTimeInterval:2];
NSLog(@"end任务1");
});
dispatch_group_async(group, queue, ^{
NSLog(@"star任务2");
[NSThread sleepForTimeInterval:2];
NSLog(@"end任务2");
});
dispatch_group_async(group, queue, ^{
NSLog(@"star任务3");
[NSThread sleepForTimeInterval:2];
NSLog(@"end任务3");
});
dispatch_group_notify(group, queue, ^{
NSLog(@"所有任务执行完毕的回调");
dispatch_async(dispatch_get_main_queue(), ^{
NSLog(@"回到主线程刷新UI");
});
});
观察线程id,3个任务在3个子线程;所有任务完成的通知也是在其中一个子线程;主线程id是18153828
。
假如我们有2个异步的请求同时发送
- (void)sendRequest1:(void(^)())block
{
dispatch_async(dispatch_get_global_queue(0, 0), ^{
NSLog(@"star任务1");
[NSThread sleepForTimeInterval:2];
NSLog(@"end任务1");
// 异步任务完成之后,回到主线程
dispatch_async(dispatch_get_main_queue(), ^{
if (block) {
block();
}
});
});
}
- (void)sendRequest2:(void(^)())block
{
dispatch_async(dispatch_get_global_queue(0, 0), ^{
NSLog(@"star任务2");
[NSThread sleepForTimeInterval:2];
NSLog(@"end任务2");
// 异步任务完成之后,回到主线程
dispatch_async(dispatch_get_main_queue(), ^{
if (block) {
block();
}
});
});
}
上面2个异步网络请求方法,我们在dispatch_group_async
使用:
// 创建一个并行的queue
dispatch_queue_t queue = dispatch_queue_create("gcd.group", DISPATCH_QUEUE_CONCURRENT);
// 创建group
dispatch_group_t group = dispatch_group_create();
dispatch_group_async(group, queue, ^{
[self sendRequest1:^{
NSLog(@"请求1完成");
}];
});
dispatch_group_async(group, queue, ^{
[self sendRequest2:^{
NSLog(@"请求2完成");
}];
});
dispatch_group_notify(group, queue, ^{
NSLog(@"所有任务执行完毕的回调");
dispatch_async(dispatch_get_main_queue(), ^{
NSLog(@"回到主线程刷新UI");
});
});
查看控制台打印:在2个异步请求还没有结束,就调用了dispatch_group_notify
,显然这不是我们预期的结果。
那么,怎么处理这个问题呢?这就需用用到另外的方法,具体看下面代码:
// 创建一个并行的queue
dispatch_queue_t queue = dispatch_queue_create("gcd.group", DISPATCH_QUEUE_CONCURRENT);
// 创建group
dispatch_group_t group = dispatch_group_create();
dispatch_group_enter(group);
[self sendRequest1:^{
NSLog(@"请求1完成");
dispatch_group_leave(group);
}];
dispatch_group_enter(group);
[self sendRequest2:^{
NSLog(@"请求2完成");
dispatch_group_leave(group);
}];
dispatch_group_notify(group, queue, ^{
NSLog(@"所有任务执行完毕的回调");
dispatch_async(dispatch_get_main_queue(), ^{
NSLog(@"回到主线程刷新UI");
});
});
现在就是我们预期的效果了。