一个页面中有多个网络请求时,怎么写比较合理?

一、2个请求没有依赖性,但是要求2个请求都完成后才能刷新UI。

二、2个请求有依赖性,A请求完成后,B请求需要用到 A 返回的结果  作为参数进行请求。

 

先解决第一个:

使用GCD的 group ,等两个请求都完成后,在刷新UI.代码如下:

- (void)request{
    
//创建一个GCD group
    self.group = dispatch_group_create();
    
//将请求放入group中处理
    dispatch_group_async(self.group, dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
        [self request1];
    });
    
    dispatch_group_async(self.group, dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
        [self request2];
    });
    
    dispatch_group_async(self.group, dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
        [self request3];
    });
    
//监听group中任务是否都完成了,完成了就执行刷新界面
    dispatch_group_notify(self.group, dispatch_get_main_queue(), ^{
        NSLog(@"刷新界面");
    });
}
//请求1
-(void)request1{
    
    AFHTTPSessionManager *manger = [AFHTTPSessionManager manager];
    NSString *url = @"http://v3.wufazhuce.com:8000/api/channel/movie/more/0?platform=ios&version=v4.0.1";
    [manger GET:url parameters:nil progress:^(NSProgress * _Nonnull downloadProgress) {
        
    } success:^(NSURLSessionDataTask * _Nonnull task, id  _Nullable responseObject) {
        NSArray *data = responseObject[@"data"];
        for (NSDictionary *dic in data) {
            NSLog(@"请求1---%@",dic[@"id"]);
        }
        

    } failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) {

    }];
}



//请求2
-(void)request2{
    
    AFHTTPSessionManager *manger = [AFHTTPSessionManager manager];
    NSString *url = @"http://v3.wufazhuce.com:8000/api/channel/movie/more/0?platform=ios&version=v4.0.1";
    [manger GET:url parameters:nil progress:^(NSProgress * _Nonnull downloadProgress) {
        
    } success:^(NSURLSessionDataTask * _Nonnull task, id  _Nullable responseObject) {
        NSArray *data = responseObject[@"data"];
        for (NSDictionary *dic in data) {
            NSLog(@"请求2---%@",dic[@"id"]);
        }
        

    } failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) {
        

    }];
}



//请求3
-(void)request3{
    
    AFHTTPSessionManager *manger = [AFHTTPSessionManager manager];
    NSString *url = @"http://v3.wufazhuce.com:8000/api/channel/movie/more/0?platform=ios&version=v4.0.1";
    [manger GET:url parameters:nil progress:^(NSProgress * _Nonnull downloadProgress) {
        
    } success:^(NSURLSessionDataTask * _Nonnull task, id  _Nullable responseObject) {
        NSArray *data = responseObject[@"data"];
        for (NSDictionary *dic in data) {
            NSLog(@"请求3---%@",dic[@"id"]);
        }
        

    } failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) {
        

    }];
}

运行结果:

064933_ob3W_2472425.png

我擦,说好的先执行group中的任务,等group中的任务都执行完成后,再执行刷新UI的呢?事与愿违啊!

经过查证找出了原因,原因就是:我们的网络请求AFN发送请求也是在异步线程的。上面这种将请求放入group的写法遵守的整组全部执行完毕指的是三个block写的代码直观进行执行完,就算执行完毕。也就是说[self request1] 执行AFN的代码完了,不包含AFN异步请求。  因此会出现这样的结果。

 

改正:

网上有很多人用group 和 信号量来解决这个问题,其实不需要信号量,队列组就能解决了。只要换一种方式将任务加入队列组就行了。

- (void)request{
    
    self.group = dispatch_group_create();
    
    //下面的任务马上要放到group中执行了
    dispatch_group_enter(self.group);
    [self request1];
    
    
    //下面的任务马上要放到group中执行了
    dispatch_group_enter(self.group);
    [self request2];
    
    
    //下面的任务马上要放到group中执行了
    dispatch_group_enter(self.group);
    [self request3];
    
    
    dispatch_group_notify(self.group, dispatch_get_main_queue(), ^{
        NSLog(@"刷新界面");
    });
}
-(void)request1{
    
    
    AFHTTPSessionManager *manger = [AFHTTPSessionManager manager];
    NSString *url = @"http://v3.wufazhuce.com:8000/api/channel/movie/more/0?platform=ios&version=v4.0.1";
    [manger GET:url parameters:nil progress:^(NSProgress * _Nonnull downloadProgress) {
        
    } success:^(NSURLSessionDataTask * _Nonnull task, id  _Nullable responseObject) {
        NSArray *data = responseObject[@"data"];
        for (NSDictionary *dic in data) {
            NSLog(@"请求1---%@",dic[@"id"]);
        }
        //任务完成了,该任务从group中移除
        dispatch_group_leave(self.group);
        
    } failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) {
        
        //任务完成了,该任务从group中移除
        dispatch_group_leave(self.group);
    }];
}




-(void)request2{
    
   
    AFHTTPSessionManager *manger = [AFHTTPSessionManager manager];
    NSString *url = @"http://v3.wufazhuce.com:8000/api/channel/movie/more/0?platform=ios&version=v4.0.1";
    [manger GET:url parameters:nil progress:^(NSProgress * _Nonnull downloadProgress) {
        
    } success:^(NSURLSessionDataTask * _Nonnull task, id  _Nullable responseObject) {
        NSArray *data = responseObject[@"data"];
        for (NSDictionary *dic in data) {
            NSLog(@"请求2---%@",dic[@"id"]);
        }
        //任务完成了,该任务从group中移除
        dispatch_group_leave(self.group);
        
    } failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) {
        
        //任务完成了,该任务从group中移除
        dispatch_group_leave(self.group);
    }];
}



-(void)request3{
    
    
    AFHTTPSessionManager *manger = [AFHTTPSessionManager manager];
    NSString *url = @"http://v3.wufazhuce.com:8000/api/channel/movie/more/0?platform=ios&version=v4.0.1";
    [manger GET:url parameters:nil progress:^(NSProgress * _Nonnull downloadProgress) {
        
    } success:^(NSURLSessionDataTask * _Nonnull task, id  _Nullable responseObject) {
        NSArray *data = responseObject[@"data"];
        for (NSDictionary *dic in data) {
            NSLog(@"请求3---%@",dic[@"id"]);
        }
        
        //任务完成了,该任务从group中移除
        dispatch_group_leave(self.group);
        
    } failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) {
        
        //任务完成了,该任务从group中移除
        dispatch_group_leave(self.group);
    }];
}

这种dispatch_group_enterdispatch_group_leave是成对出现的,当进入一个操作后,在适当的位置加上dispatch_group_leave,比如在这里,进入第一个 enter 执行 request1 操作,那么在这个方法中异步返回的时候(成功或失败)写上 dispatch_gropu_leave,这样就包含了异步的情况。

 

 

二、解决有依赖的两个请求

// 创建一个队列
    NSOperationQueue *queue = [[NSOperationQueue alloc]init];
    
    
    // 创建一个A操作
    NSBlockOperation *operationA = [NSBlockOperation blockOperationWithBlock:^{

        
        [self requestA];
    }];
    
    
    
    // 创建一个B操作
    NSBlockOperation *operationB = [NSBlockOperation blockOperationWithBlock:^{

        [self requestB];
        
    }];
    
    
    //添加依赖 B要在A打印完在进行打印 所以是B依赖于A 那么只需要添加如下代码即可完成
    [operationB addDependency:operationA];
    
    
    // 分别加入到队列中
    [queue addOperation:operationA];
    [queue addOperation:operationB];
-(void)requestA{
    
    
    AFHTTPSessionManager *manger = [AFHTTPSessionManager manager];
    NSString *url = @"http://v3.wufazhuce.com:8000/api/channel/movie/more/0?platform=ios&version=v4.0.1";
    [manger GET:url parameters:nil progress:^(NSProgress * _Nonnull downloadProgress) {
        
    } success:^(NSURLSessionDataTask * _Nonnull task, id  _Nullable responseObject) {
        NSArray *data = responseObject[@"data"];
        for (NSDictionary *dic in data) {
            NSLog(@"请求AAA---%@",dic[@"id"]);
        }
        
        
    } failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) {
        
    }];
}


-(void)requestB{
    
    
    AFHTTPSessionManager *manger = [AFHTTPSessionManager manager];
    NSString *url = @"http://v3.wufazhuce.com:8000/api/channel/movie/more/0?platform=ios&version=v4.0.1";
    [manger GET:url parameters:nil progress:^(NSProgress * _Nonnull downloadProgress) {
        
    } success:^(NSURLSessionDataTask * _Nonnull task, id  _Nullable responseObject) {
        NSArray *data = responseObject[@"data"];
        for (NSDictionary *dic in data) {
            NSLog(@"请求BBB---%@",dic[@"id"]);
        }
        
        
    } failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) {
        
    }];
}

这里我没有在一个线程请求返回成功后,再发起第二个请求,这种方式也能解决问题,但是这样写代码会比较乱,我不喜欢这么写。

转载于:https://my.oschina.net/u/2472425/blog/1353216

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值