GCD-两个网络请求同步问题

在网络请求的时候有时有这种需求

两个接口请求数据,然后我们才能做最后的数据处理。但是因为网络请求是移步的 。我们并不知道什么时候两个请求完成 。

通常面对这样的需求会自然的想到 多线程 啊 。表现真正的技术的时刻来啦,可以使用 group 队列啊 。等队列中的请求任务都完成 ,在通知主线程处理汇总数据嘛 。

今天我也是这么写的,但是发现主线程并没有等到队列中的分线程网络请求bock回调就返回了 。我给block回调之前打印,确实是队列中的任务都打印之后,才返回的主线程 。那么问题在哪里 ?



网络请求然后处理响应数据是个耗时的操作,也是我们开发中常见的一种情形,在网络请求以及处理响应数据操作完毕之后我们在执行别的操作这样的过程也是我们开发中常见的情形。我们可以知道,

网络请求的任务是提交给子线程异步处理了,网络请求这样的任务也就快速执行完毕了,但是网络请求是一个任务,处理收到的网络响应又是一个任务,注意不要把这两个过程混为一谈

而收到网络响应以及处理返回响应的数据并不是在子线程中执行的,我们通过在回调响应处理的block中打印当前线程,会发现回调响应处理的block是在主线程中被执行的。

如果很熟悉block回调这种通信机制的话,就不难理解,这个回调响应的block真正被调用执行的地方应该是AFN框架的底层代码,而这部分代码显然是在主线程中执行的。

这时候,如果我们需要确定这个主线程中收到网络响应的数据被处理操作结束之后,才最后执行我们需要最后的操作。换句话说,自线程就要等待,收到一个信号,才通知主线程,自己真正的完成任务了 。

这个信号就是GCD的信号量 dispatch_semaphore_t

- (void)getNetworkingData{
     NSString *appIdKey = @"8781e4ef1c73ff20a180d3d7a42a8c04";
     NSString* urlString_1 = @"http://api.openweathermap.org/data/2.5/weather";
     NSString* urlString_2 = @"http://api.openweathermap.org/data/2.5/forecast/daily";
     NSDictionary* dictionary [email protected]{@"lat":@"40.04991291",
                                 @"lon":@"116.25626162",
                                 @"APPID" : appIdKey};
     // 创建组
     dispatch_group_t group = dispatch_group_create();
     // 将第一个网络请求任务添加到组中
     dispatch_group_async(group, dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
         // 创建信号量
         dispatch_semaphore_t semaphore = dispatch_semaphore_create(0);
         // 开始网络请求任务
         AFHTTPSessionManager *manager = [AFHTTPSessionManager manager];
         [manager GET:urlString_1
           parameters:dictionary
             progress:nil
              success:^(NSURLSessionDataTask * _Nonnull task, id  _Nullable responseObject) {
                  NSLog(@"成功请求数据1:%@",[responseObject class]);
                  // 如果请求成功,发送信号量
                  dispatch_semaphore_signal(semaphore);
              } failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) {
                  NSLog(@"失败请求数据");
                  // 如果请求失败,也发送信号量
                  dispatch_semaphore_signal(semaphore);
              }];
         // 在网络请求任务成功之前,信号量等待中
         dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER);
     });
     // 将第二个网络请求任务添加到组中
     dispatch_group_async(group, dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
         // 创建信号量
         dispatch_semaphore_t semaphore = dispatch_semaphore_create(0);
         // 开始网络请求任务
         AFHTTPSessionManager *manager = [AFHTTPSessionManager manager];
         [manager GET:urlString_2
           parameters:dictionary
             progress:nil
              success:^(NSURLSessionDataTask * _Nonnull task, id  _Nullable responseObject) {
                  NSLog(@"成功请求数据2:%@",[responseObject class]);
                  // 如果请求成功,发送信号量
                  dispatch_semaphore_signal(semaphore);
              } failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) {
                  NSLog(@"失败请求数据");
                  // 如果请求失败,也发送信号量
                  dispatch_semaphore_signal(semaphore);
              }];
         // 在网络请求任务成功之前,信号量等待中
         dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER);
     });
     dispatch_group_notify(group, dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
         NSLog(@"完成了网络请求,不管网络请求失败了还是成功了。");
     });
 }

这样做的具体步骤是这样的 。在自线程队列中 。设置的信号等待 ,一直到block回调完成(主线程中),发送信号 。子线程收到信号,然后才会通知dispatch_group_notify 子线程的请求数据真正返回了。

在使用的时候一定要想清楚哪个需要等待,哪个线程来发送。

文/苏永茂(简书作者)
原文链接:http://www.jianshu.com/p/943dcb9ad632
著作权归作者所有,转载请联系作者获得授权,并标注“简书作者”。

时间: 2024-10-22 01:52:35

GCD-两个网络请求同步问题的相关文章

IOS网络请求(同步GET,同步POST,异步GET,异步POST)

1.     同步GET请求     //第一步,创建URL     NSURL *url = [NSURL URLWithString:@"http://api.hudong.com/iphonexml.do?type=focus-c"];          //第二步,通过URL创建网络请求     NSURLRequest *request = [[NSURLRequest alloc]initWithURL:url cachePolicy:NSURLRequestUseProt

GCD实现多个网络请求同步执行

1.dispatch_group_async.dispatch_group_t与dispatch_group_notify 再配合dispatch_group_enter(group)和dispatch_group_leave(group)两个函数一起来使用,这样才能实现我们想要的最终效果. 2.dispatch_group_enter(dispatch_group_t group); 参数group不能为空,在异步任务开始前调用.它明确的表明了一个block被加入到了队列组group中,此时g

IOS ---两种网络请求的最佳实践

在使用URL 加载系统时,主要的两种方法可以执行HTTP请求和接收响应: 一.同步请求的最佳实践 只在后台线程中使用同步请求,除非确定请求访问的是本地文件资源,否则请不要再主线程上使用: 只有在知道返回的数据不会超出应用的内存时才使用同步请求.记住,整个响应体都会在位于代码的内存中.如果响应很大,那么可能导致应用出现内存溢出问题.此外,当代吗将响应解析为所需的格式时可能需要复制返回的数据,这会导致内存增加一倍: 在处理返回的数据前,验证错误与调用返回的HTTP响应状态码: 如果源URL需要验证,

(转载)Android之三种网络请求解析数据(最佳案例)

[置顶] Android之三种网络请求解析数据(最佳案例) 2016-07-25 18:02 4725人阅读 评论(0) 收藏 举报  分类: Gson.Gson解析(1)  版权声明:本文为博主原创文章,未经博主允许不得转载. 目录(?)[+] 小武:相信大家都用过网络请求解析数据,只是方法不一样而已,但是,逻辑都是差不多的: 一:AsyncTask解析数据 AsyncTask主要用来更新UI线程,比较耗时的操作可以在AsyncTask中使用. AsyncTask是个抽象类,使用时需要继承这个

网络请求框架---Volley

去年的Google I/O大会为android开发者带来了一个网络请求框架,它的名字叫做Volley.Volley诞生的使命就是让Android的网络请求更快,更健壮,而且它的网络通信的实现是基于HttpURLConnection的.前几天我整理了两个网络请求工具类,应付日常工作的开发需求还是没问题的,自己分别在HttpURLConnection和HttpClient的基础上进行进一层的封装,保证了API的足够简洁友好.有兴趣的同学可以回头去翻翻,源码在Github上,搜索AndroidHttp

Android 网络请求详解

我们知道大多数的 Android 应用程序都是通过和服务器进行交互来获取数据的.如果使用 HTTP 协议来发送和接收网络数据,就免不了使用 HttpURLConnection 和 HttpClient,而 Android 中主要提供了上述两种方式来进行 HTTP 操作.并且这两种方式都支持 HTTPS 协议.以流的形式进行上传和下载.配置超时时间.IPv6.以及连接池等功能. 但是 Googl e发布 6.0 版本的时候声明原生剔除 HttpClient,但是笔者认为 HttpClient 会提

iOS网络编程同步GET方法请求编程

iOS SDK为HTTP请求提供了同步和异步请求两种不同的API,而且可以使用GET或POST等请求方法.我们先了解其中最为简单的同步GET方法请求. 首先实现查询业务,查询业务请求可以在主视图控制器MasterViewController类中实现,其中MasterViewController.h代码如下: Java代码   #import <UIKit/UIKit.h> #import “NSString+URLEncoding.h” #import “NSNumber+Message.h”

iOS 网络与多线程--4.同步Post方式的网络请求

通过Post请求方式,同步获取网络数据,一旦发送同步请求,程序将停止用户交互,直至服务器返回数据 在ViewController.m文件内的viewDidLoad函数添加一下测试代码 1 - (void)viewDidLoad { 2 [super viewDidLoad]; 3 // Do any additional setup after loading the view, typically from a nib. 4 5 // 1.建立一个网址对象,指定请求数据的网址 6 NSURL

Android下网络请求的两个封装类

1.背景介绍 我们在Activity和Fragment中做网络请求的时候,往往用到线程和handler消息处理机制等做处理.这里,我们将网络请求,一个get和一个post请求,使用线程封装一下.这样就剥离开了Activity中的代码量,将复杂的代码,封装在不同的类中,实现一个类的单一原则. 2.GET请求封装 /** * GET提交的网络请求<br> * <br> * 传递过来主UI线程的消息处理器,经过线程处理之后,返回一个消息给主UI线程 <br> * 在返回的ms