dispatch_group的实际使用案例,监听多任务

版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/shaohua_lv/article/details/77186133

 在发起网络请求时,我们一般会用异步请求,这里我们以 AFNetWorking 为例:

AFHTTPRequestOperationManager *manager = [AFHTTPRequestOperationManager manager];  
[manager GET:@"http://octree.me/" parameters:nil success:^(AFHTTPRequestOperation *operation, id responseObject) {
    NSLog(@"Success");
} failure:^(AFHTTPRequestOperation *operation, NSError *error) {
    NSLog(@"Error: %@", error);
}];

 假如我们要执行多个异步请求,一般可以这么写

NSArray *urlStrings = @[ @"http://octree.me", @"http://google.com", @"http://github.com" ];

for(NSString urlString in urlStrings) {  
    AFHTTPRequestOperationManager *manager = [AFHTTPRequestOperationManager manager];
    [manager GET:urlString parameters:nil success:^(AFHTTPRequestOperation *operation, id responseObject) {
        NSLog(@"Success");
     } failure:^(AFHTTPRequestOperation *operation, NSError *error) {
        NSLog(@"Error: %@", error);
}];
}

  假如我们需要在三个请求都完成或者失败后进行一些处理,但是 manager 发起的请求时异步处理的,也就是说当 manager 调用 GET:parameters:success:failure 会立即返回,当请求成功或失败后才会调用各自的 block ,我们如何才能监控并发的异步事件呢?并且他们的完成顺序以及完成时间都是不确定的。

  当然,你可以用一组 bool 值或者其他的标识记录每个任务的完成进度,但是这样的代码不仅丑陋,而且失去了扩展性。

  这时候我们就可以用到 dispatch_group 了

dispatch_group 在任务组内的任务都完成的时候通过同步或者异步的方式通知你

dispatch_group 提供了两种通知方式, dispatch_group_wait 和 dispatch_group_notify

dispatch_group_wait 会阻塞当前线程,知道任务都完成时才会继续执行下面的代码

  我们可以使用 dispatch_group_wait 这样实现:

NSArray *urlStrings = @[ @"http://octree.me", @"http://google.com", @"http://github.com" ];

dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_HIGH, 0), ^{

    dispatch_group_t requestGroup = dispatch_group_create();

    for(NSString urlString in urlStrings) {
    dispatch_group_enter(requestGroup);
    AFHTTPRequestOperationManager *manager = [AFHTTPRequestOperationManager manager];
    [manager GET:urlString parameters:nil success:^(AFHTTPRequestOperation *operation, id responseObject) {

        NSLog(@"Success");
        dispatch_group_leave(requestGroup);
     } failure:^(AFHTTPRequestOperation *operation, NSError *error) {

        NSLog(@"Error: %@", error);
        dispatch_group_leave(requestGroup);
    }];
    }

    dispatch_group_wait(requestGroup, DISPATCH_TIME_FOREVER);
    dispatch_async(dispatch_get_main_queue(), ^{
            //doSomething;
        });
}];

  因为 dispatch_group_wait 会阻塞当前线程,所以我们要把他放到后台执行,避免阻塞主线程。通过 dispatch_group_enter 和 dispatch_group_leave 手动通知任务的开始以及结束。 dispatch_group_wait 会阻塞当前线程,知道所有任务完成或者超时才会继续执行下面的代码。

  前面我们说过, dispatch_group 提供了两种通知方式,我们已经了解了 dispatch_group_wait ,另一种是 dispath_group_notify ,这种方式相对于前面的显得更为灵活。

  dispatch_group_notify 是通过异步的方式通知,所以,不会阻塞线程。于是,我们就可以这样写:

NSArray *urlStrings = @[ @"http://octree.me", @"http://google.com", @"http://github.com" ];

dispatch_group_t requestGroup = dispatch_group_create();

for(NSString urlString in urlStrings) {  
    dispatch_group_enter(requestGroup);
    AFHTTPRequestOperationManager *manager = [AFHTTPRequestOperationManager manager];
    [manager GET:urlString parameters:nil success:^(AFHTTPRequestOperation *operation, id responseObject) {

        NSLog(@"Success");
        dispatch_group_leave(requestGroup);
     } failure:^(AFHTTPRequestOperation *operation, NSError *error) {

        NSLog(@"Error: %@", error);
        dispatch_group_leave(requestGroup);
    }];
}

dispatch_group_notify(requestGroup, dispatch_get_main_queue(), ^{

        //doSomething
    });
展开阅读全文

没有更多推荐了,返回首页