iOS 控制任务执行顺序
有两个任务都要放在子线程中执行, 且要保证a执行完再执行b.
如果你也有同样的需求, 可以从下面几种方法中挑选一种
实现方法
1.1. 方法一 使用gcd串行队列. 先创建一个串行队列, 将任务加入到串行队列中, 因为是串行队列,能保证顺序执行, 且因为是串行队列,所以两个任务是在同一线程执行.
dispatch_queue_t queue = dispatch_queue_create("mycostom", dispatch_queue_serial); nslog(@"dispatch111 start"); dispatch_async(queue, ^{ nslog(@"sleep1111"); nslog(@"dispatch111 %@", [nsthread currentthread]); [nsthread sleepfortimeinterval:5]; nslog(@"awake up111"); }); nslog(@"dispatch2222 start"); dispatch_async(queue, ^{ nslog(@"sleep 2222"); nslog(@"dispatch2222 %@", [nsthread currentthread]); [nsthread sleepfortimeinterval:5]; nslog(@"awake up222222"); }); nslog(@"dispatch 33333");
1.2 方法二 使用barrier.
dispatch_queue_t queue = dispatch_queue_create("mycustom", dispatch_queue_concurrent); nslog(@"dispatch111 start"); dispatch_async(queue, ^{ nslog(@"sleep1111"); nslog(@"dispatch111 %@", [nsthread currentthread]); [nsthread sleepfortimeinterval:5]; nslog(@"awake up111"); }); dispatch_barrier_async(queue, ^{ nslog(@"------------------- barrier"); }); nslog(@"dispatch2222 start"); dispatch_async(queue, ^{ nslog(@"sleep 2222"); nslog(@"dispatch2222 %@", [nsthread currentthread]); [nsthread sleepfortimeinterval:5]; nslog(@"awake up222222"); }); nslog(@"dispatch 33333");
打印结果
2017-07-18 16:42:48.866 test[4336:640379] dispatch111 start 2017-07-18 16:42:48.866 test[4336:640379] dispatch2222 start 2017-07-18 16:42:48.866 test[4336:640550] sleep1111 2017-07-18 16:42:48.866 test[4336:640379] dispatch 33333 2017-07-18 16:42:48.866 test[4336:640550] dispatch111 {number = 3, name = (null)} 2017-07-18 16:42:53.869 test[4336:640550] awake up111 2017-07-18 16:42:53.870 test[4336:640550] ------------------- barrier 2017-07-18 16:42:53.870 test[4336:640550] sleep 2222 2017-07-18 16:42:53.870 test[4336:640550] dispatch2222 {number = 3, name = (null)} 2017-07-18 16:42:58.872 test[4336:640550] awake up222222
^^^^
重点 你可以观察到 先打印了dispatch 333 在打印 barrier, 说明barrier 阻塞的不是barrier所在的线程,而是queue所在的线程
重点如果我把 dispatch_barrier_async 换成
dispatch_barrier_sync 这就连当前线程也阻塞了
1.3 方法三 使用nsoperation
在这之前简单介绍一下 nsoperation
* nsoperation 是对gcd的封装.
* nsoperation 是一个抽象类, 不能封装任务, 需要他的两个子类完成 nsinvocationoperation 和
nsblockoperation
* 需要调用start 方法开启, 默认是同步执行,也就是说默认不会开线程, 也可以使用 cancel 方法取消, 但是只能取消未开始的任务,已经开始的任务,无法被取消.
* 在nsoperation 添加到queue后, 通常会在很短的时间内得到运行, 但是如果operation有添加依赖或者queue 被设置成等待, 那可能需要等待.
结合最后一条, 所以我们现在来使用依赖实现我们的需求.
nsoperationqueue * queue = [[nsoperationqueue alloc] init]; nsblockoperation * op1 = [nsblockoperation blockoperationwithblock:^{ nslog(@"sleep1111"); nslog(@"dispatch111 %@", [nsthread currentthread]); [nsthread sleepfortimeinterval:5]; nslog(@"awake up111"); }]; nsblockoperation * op2 = [nsblockoperation blockoperationwithblock:^{ nslog(@"sleep 2222"); nslog(@"dispatch2222 %@", [nsthread currentthread]); [nsthread sleepfortimeinterval:5]; nslog(@"awake up222222"); }]; [op1 adddependency:op2]; [queue addoperation:op1]; [queue addoperation:op2];
1.4 方法四, 既然nsoperation是对gcd的封装,那么联想方法一, 我想能不能将nsoperationqueue也设置成串行队列.
但是nsoperationqueue好像屏蔽了这个串行并行这个概念, 而是使用 queue.maxconcurrentoperationcount 来设置queue一次能够执行的任务量.
如果设置成1. 就类似于串行队列, 任务会 顺序执行 . 但是只是类似, 因为在串行队列 + 异步执行时,会开辟一条子线程. 但是将maxconcurrentoperationcount 设置成1后, 有可能会开辟多条线程,也就是说, 任务一和任务二可能会在同一线程上执行, 也可能在不同线程上执行.
nsoperationqueue *queue = [[nsoperationqueue alloc] init]; queue.maxconcurrentoperationcount = 1; nslog(@"nsoperation111 start"); nsblockoperation *operationb = [nsblockoperation blockoperationwithblock:^{ nslog(@"sleep1111"); nslog(@"nsoperation111 %@", [nsthread currentthread]); [nsthread sleepfortimeinterval:3]; nslog(@"awake up111"); }]; nslog(@"nsoperation2222 start"); nsinvocationoperation *operationi = [[nsinvocationoperation alloc] initwithtarget:self selector:@selector(testinvocaionoperatation) object:nil]; [queue addoperation:operationb]; [queue addoperation:operationi]; nslog(@"nsoperation 33333")
打印结果
2017-07-18 19:13:56.971 test[4691:708449] nsoperation111 start 2017-07-18 19:13:56.972 test[4691:708449] nsoperation2222 start 2017-07-18 19:13:56.972 test[4691:708449] nsoperation 33333 2017-07-18 19:13:56.972 test[4691:708483] sleep1111 2017-07-18 19:13:56.973 test[4691:708483] nsoperation111 {number = 3, name = (null)} 2017-07-18 19:14:00.048 test[4691:708483] awake up111 2017-07-18 19:14:00.049 test[4691:708481] sleep 2222 2017-07-18 19:14:00.049 test[4691:708481] nsoperation2222 {number = 4, name = (null)} 2017-07-18 19:14:03.110 test[4691:708481] awake up222222
观察可知, 当前两个任务没有在同一线程上执行,但是sleep222 和awake up222都是在 awake111后执行的, 也即是说是顺序执行。
上一篇: 判断是否是手机
下一篇: “限韩令”有望解禁 腾讯“绝地求生”