iOS 通过添加线程依赖和信号量结合实现一个复杂界面请求多个接口时按指定顺序执行
2017-12-09 21:16
731 查看
前言:本文通过作者的一种思想实现关于iOS中一个界面请求多条数据时,需要按指定顺序执行
主要思路:为了实现多个请求接口按指定顺序前后执行,本文主要通过添加线程依赖关系和GCD的信号量相结合来实现基本需求贴上Demo地址,感觉可以的麻烦记得点个星星。直接上代码
线程依赖关系通过使用系统对GCD的进一步封装的类NSBlockOperation来实现
-(void)dispatchAllRequest{ // 利用线程依赖关系测试 __weak typeof (self)weakSelf =self; NSBlockOperation * operation1 = [NSBlockOperation blockOperationWithBlock:^{ [weakSelf requestA]; }]; NSBlockOperation * operation2 = [NSBlockOperation blockOperationWithBlock:^{ [weakSelf requestB]; }]; NSBlockOperation * operation3 = [NSBlockOperation blockOperationWithBlock:^{ [weakSelf requestC]; }]; [operation2 addDependency:operation1]; [operation3 addDependency:operation2]; NSOperationQueue * queue = [[NSOperationQueue alloc]init]; [queue addOperations:@[operation1,operation2,operation3] waitUntilFinished:NO]; }
请求方法的话使用GCD的信号量来实现阻塞线程来达到我们线程同步请求的目的
第一个请求数据方法
-(void)requestA{ //创建信号量并设置计数默认为0 dispatch_semaphore_t sema = dispatch_semaphore_create(0); AFHTTPSessionManager *manager = [AFHTTPSessionManager manager];; manager.responseSerializer = [AFHTTPResponseSerializer serializer]; manager.responseSerializer.acceptableContentTypes = [NSSet setWithObjects:@"application/json", @"text/json", @"text/javascript",@"text/html",nil]; [manager GET:@"http://qr.bookln.cn/qr.html?crcode=110000000F00000000000000B3ZX1CEC" parameters:nil progress:nil success:^(NSURLSessionDataTask * _Nonnull task, id _Nullable responseObject) { dispatch_semaphore_signal(sema); NSLog(@"正在执行A"); } failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) { ////计数+1操作 dispatch_semaphore_signal(sema); NSLog(@"执行错误A"); }]; NSLog(@"正在刷新A"); //若计数为0则一直等待 dispatch_semaphore_wait(sema, DISPATCH_TIME_FOREVER); NSLog(@"已经刷新A"); }
第二个请求数据方法
-(void)requestB{ //创建信号量并设置计数默认为0 dispatch_semaphore_t sema = dispatch_semaphore_create(0); AFHTTPSessionManager *manager = [AFHTTPSessionManager manager];; manager.responseSerializer = [AFHTTPResponseSerializer serializer]; manager.responseSerializer.acceptableContentTypes = [NSSet setWithObjects:@"application/json", @"text/json", @"text/javascript",@"text/html",nil]; [manager GET:@"http://qr.bookln.cn/qr.html?crcode=110000000F00000000000000B3ZX1CEC" parameters:nil progress:nil success:^(NSURLSessionDataTask * _Nonnull task, id _Nullable responseObject) { dispatch_semaphore_signal(sema); NSLog(@"正在执行B"); } failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) { ////计数+1操作 dispatch_semaphore_signal(sema); NSLog(@"执行错误B"); }]; NSLog(@"正在刷新B"); //若计数为0则一直等待 dispatch_semaphore_wait(sema, DISPATCH_TIME_FOREVER); NSLog(@"已经刷新B"); }
第三个请求数据方法
-(void)requestC{ //创建信号量并设置计数默认为0 dispatch_semaphore_t sema = dispatch_semaphore_create(0); AFHTTPSessionManager *manager = [AFHTTPSessionManager manager];; manager.responseSerializer = [AFHTTPResponseSerializer serializer]; manager.responseSerializer.acceptableContentTypes = [NSSet setWithObjects:@"application/json", @"text/json", @"text/javascript",@"text/html",nil]; [manager GET:@"http://qr.bookln.cn/qr.html?crcode=110000000F00000000000000B3ZX1CEC" parameters:nil progress:nil success:^(NSURLSessionDataTask * _Nonnull task, id _Nullable responseObject) { dispatch_semaphore_signal(sema); NSLog(@"正在执行C"); } failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) { ////计数+1操作 dispatch_semaphore_signal(sema); NSLog(@"执行错误C"); }]; NSLog(@"正在刷新C"); //若计数为0则一直等待 dispatch_semaphore_wait(sema, DISPATCH_TIME_FOREVER); NSLog(@"已经刷新C"); }
为了模拟现实项目中的请求过程,所以用到了请求接口三方AFNetworking
更加贴近真实效果,亲测对于一个界面多个接口请求数据需要同步请求的是可用的。最后附上GCD的信号量的一些小知识
关于信号量 信号量:就是一种可用来控制访问资源的数量的标识,设定了一个信号量,在线程访问之前,加上信号量的处理,则可告知系统按照我们指定的信号量数量来执行多个线程。 其实,这有点类似锁机制了,只不过信号量都是系统帮助我们处理了,我们只需要在执行线程之前,设定一个信号量值,并且在使用时,加上信号量处理方法就行了。 信号量为0则阻塞线程,大于0则不会阻塞。因此我们可以通过改变信号量的值,来控制是否阻塞线程,从而达到线程同步。 在GCD中有三个函数是semaphore的操作,分别是: dispatch_semaphore_create 创建一个semaphore dispatch_semaphore_signal 发送一个信号 dispatch_semaphore_wait 等待信号 简单的介绍一下这三个函数,第一个函数有一个整形的参数,我们可以理解为信号的总量,dispatch_semaphore_signal是发送一个信号,自然会让信号总量加1,dispatch_semaphore_wait等待信号,当信号总量少于0的时候就会一直等待,否则就可以正常的执行,并让信号总量-1,根据这样的原理,我们便可以快速的创建一个并发控制来同步任务和有限资源访问控制。
关于NSOperationQueue
创建队列的两种方式 1.主队列 凡是添加到主队列的任务(NSOperation),都会放在主线程执行 NSOperationQueue *queue = [NSOperationQueue mainQueue]; 2.其他队列(非主队列) 添加到这种队列的任务(NSOperation),就会自动放到子现场中执行 同时包含了:串形,并发功能 NSOperationQueue *queue = [[NSOperationQueue alloc] init]; 以下是系统API提供的添加任务到队列中的方式 - (void)addOperation:(NSOperation *)op; - (void)addOperations:(NSArray<NSOperation *> *)ops waitUntilFinished:(BOOL)wait NS_AVAILABLE(10_6, 4_0); 以下API就是添加线程依赖关系 - (void)addDependency:(NSOperation *)op;
相关文章推荐
- iOS 解决网络请求和界面刷新顺序问题(GCD信号量实现)
- 使用信号量来实现线程的先后执行顺序
- 使用XIB自定义一个UIView,然后将这个view添加到controller的view 上(相当于所有界面都通过xib来实现)
- 使用XIB自定义一个UIView,然后将这个view添加到controller的view 上(相当于所有界面都通过xib来实现)
- 通过Timer和Handler(os)结合实现线程改变IU主界面数据
- iOS 关于dispatch_semaphore_t(信号量) 和 dispatch_group_t (组) 线程执行顺序的使用
- 通过AEC解读WF的核心原理(五)实现一个从下向上执行的顺序容器
- 信号量(随意指定线程执行顺序)
- iOS开发——使用代理(Delegate)实现跨界面执行跳转请求
- 使用XIB自定义一个UIView,然后将这个view添加到controller的view 上(相当于所有界面都通过xib来实现)
- autofac 一个接口多个实现的顺序执行
- 用VB写的一个组件,实现添加系统用户,并添加到指定组
- 用VB写的一个组件,实现添加系统用户,并添加到指定组
- 在一个界面中有一个对话条,其中有几个控件,在另外一个对话条中也有几个控件,现在通过按TAB键希望能使这两个对话条中的控件能顺序获得输入焦点
- java中的通过继承Thread和实现Runnable接口实现的线程
- windows环境下通过添加静态路由信息,实现多网卡环境下,指定某网络由某网卡来连接
- IOS学习笔记32—使用Storyboard实现复杂界面
- 启动线程的五种方式方法(通过继承Thread类或实现Runnable接口)
- IOS学习笔记32—使用Storyboard实现复杂界面
- asp.net通过实现IHttpHandler接口方法给图片添加水印图片(动态添加,不破坏原图片,但是耗资源)