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

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

通常面对这样的需求会自然的想到 多线程 啊 。表现真正的技术的时刻来啦,可以使用 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 =@{@"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
著作权归作者所有,转载请联系作者获得授权,并标注“简书作者”。

最新文章

  1. gulpfile.js 合并压缩 requirejs 的配置文件
  2. csuoj 1334: 好老师
  3. CUBRID学习笔记 6 修改用户密码
  4. Android APP安全评估工具 Drozer - 安装介绍
  5. nginx配置负载均衡与反向代理
  6. 在ubuntu on windows 上安装jekyll
  7. 我的小前端 (1)—— 安卓机和ios机的区别
  8. PHP静态成员变量和非静态成员变量
  9. 简易商品信息管理系统——首个Web项目
  10. Jupyter Notebook的快捷键
  11. 【spring源码分析】IOC容器初始化(五)
  12. 第六十四天 JS基础操作
  13. Kafka作为大数据的核心技术,你了解多少?
  14. Linux上部署多个tomcat端口设置
  15. day5 大纲
  16. (转)Silverlight调用的JS方法返回对象数组的处理方法
  17. canvas 实现时钟效果
  18. REDIS 缓存的穿透,雪崩和热点key
  19. 【LeetCode】151. Reverse Words in a String
  20. php + ajax 避免重复提交

热门文章

  1. Python之路【第五篇】python基础 之初识函数(一)和文件管理
  2. java-通过JDBC操作数据库
  3. Python数据分析
  4. .Net程序员之不学Java做安卓开发:奇怪的Java语法
  5. Ubuntu系统下lnmp环境搭建和Nginx多站点配置
  6. Visual 2015创建新项,缺少ADO.NET 实体数据模型的解决方法
  7. Ajax的js库分析简化版
  8. 设置Android Studio启动时可选最近打开过的工程
  9. .NET Core与.NET Framework、Mono之间的关系
  10. 10年微软MVP路(如何成为一个MVP?)