iOS中多线程实现方案
2016-04-16 16:57
363 查看
iOS中多线程实现方案
方案 | 简介 | 语言 | 生命周期 | 使用频率 |
---|---|---|---|---|
pthread | 通用多线程API 相信有其他语言开发经验的一定使用过 | C | 手动管理 | 几乎不用 |
NSThread | OC中的线程对象 | OC | 手动管理 | 偶尔使用 |
GCD | 能够充分利用设备的多核 | C | 自动管理 | 经常使用 |
NSOperation | 基于GCD,使用更加的面向对象 | OC | 手动管理 | 经常使用 |
GCD的使用方法
简单来说GCD的使用就2个步骤定制任务(需要执行的操作)
将任务添加到队列(用来存放任务,任务的执行遵循FIFO原则)
队列的类型
并发队列(允许多个任务并发执行,自动开启多个线程执行任务)全局并发队列
手动创建
串行队列(任务串行执行,一个任务执行完后,再执行下一个任务)
主队列(凡是添加到主队列中的任务都会放到主线程中执行)
手动创建
GCD执行任务的常用函数
同步方式执行任务dispatch_sync(<#dispatch_queue_t queue#>, <#^(void)block#>)
异步方式执行任务
dispatch_async(<#dispatch_queue_t queue#>, <#^(void)block#>)
同步和异步的区别
同步:只能在当前线程中执行任务,不具备开启新线程的能力异步:可以在新的线程中执行任务,具备开启新线程的能力
GCD执行任务的多种情况实例
使用异步方式创建任务,将任务放入串行队列中
- (void)asyncSerial{ //创建串行队列 dispatch_queue_t queue = dispatch_queue_create("lcs", DISPATCH_QUEUE_SERIAL); NSLog(@" %@", [NSThread currentThread]); dispatch_async(queue, ^{ for (NSInteger i = 0; i < 2; i++) { NSLog(@" %@ ======= %ld", [NSThread currentThread], i); } }); dispatch_async(queue, ^{ for (NSInteger i = 0; i < 2; i++) { NSLog(@" %@ ======= %ld", [NSThread currentThread], i); } }); dispatch_async(queue, ^{ for (NSInteger i = 0; i < 2; i++) { NSLog(@" %@ ======= %ld", [NSThread currentThread], i); } }); }
运行结果(开启一个新线程,串行执行)
2016-04-09 13:35:30.409 GCD的使用[1473:29019] <NSThread: 0x7f92fad07770>{number = 1, name = main} 2016-04-09 13:35:30.410 GCD的使用[1473:29101] <NSThread: 0x7f92fae03bf0>{number = 2, name = (null)} ======= 0 2016-04-09 13:35:30.410 GCD的使用[1473:29101] <NSThread: 0x7f92fae03bf0>{number = 2, name = (null)} ======= 1 2016-04-09 13:35:30.410 GCD的使用[1473:29101] <NSThread: 0x7f92fae03bf0>{number = 2, name = (null)} ======= 0 2016-04-09 13:35:30.410 GCD的使用[1473:29101] <NSThread: 0x7f92fae03bf0>{number = 2, name = (null)} ======= 1 2016-04-09 13:35:30.411 GCD的使用[1473:29101] <NSThread: 0x7f92fae03bf0>{number = 2, name = (null)} ======= 0 2016-04-09 13:35:30.411 GCD的使用[1473:29101] <NSThread: 0x7f92fae03bf0>{number = 2, name = (null)} ======= 1
使用异步方式创建任务,将任务放入主队列中
- (void)asyncMainQueue{ //主队列 (加入到主队列中的任务,都在主线程执行) dispatch_queue_t mainQueue = dispatch_get_main_queue(); NSLog(@" %@", [NSThread currentThread]); dispatch_async(mainQueue, ^{ for (NSInteger i = 0; i < 2; i++) { NSLog(@" %@ ======= %ld", [NSThread currentThread], i); } }); dispatch_async(mainQueue, ^{ for (NSInteger i = 0; i < 2; i++) { NSLog(@" %@ ======= %ld", [NSThread currentThread], i); } }); dispatch_async(mainQueue, ^{ for (NSInteger i = 0; i < 2; i++) { NSLog(@" %@ ======= %ld", [NSThread currentThread], i); } }); }
运行结果(在主队列串行执行)
2016-04-09 14:24:25.509 GCD的使用[2171:53012] {number = 1, name = main}2016-04-09 14:24:25.514 GCD的使用[2171:53012] {number = 1, name = main} ======= 0
2016-04-09 14:24:25.515 GCD的使用[2171:53012] {number = 1, name = main} ======= 1
2016-04-09 14:24:25.515 GCD的使用[2171:53012] {number = 1, name = main} ======= 0
2016-04-09 14:24:25.515 GCD的使用[2171:53012] {number = 1, name = main} ======= 1
2016-04-09 14:24:25.515 GCD的使用[2171:53012] {number = 1, name = main} ======= 0
2016-04-09 14:24:25.515 GCD的使用[2171:53012] {number = 1, name = main} ======= 1
使用异步方式创建任务,将任务放入并行队列中
- (void)asyncConcurrent{ //自己创建并发队列 dispatch_queue_t queue = dispatch_queue_create("lcs", DISPATCH_QUEUE_CONCURRENT); //全局并发队列 dispatch_queue_t globalQueue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0); NSLog(@" %@", [NSThread currentThread]); dispatch_async(queue, ^{ for (NSInteger i = 0; i < 2; i++) { NSLog(@" %@ ======= %ld", [NSThread currentThread], i); } }); dispatch_async(queue, ^{ for (NSInteger i = 0; i < 2; i++) { NSLog(@" %@ ======= %ld", [NSThread currentThread], i); } }); dispatch_async(queue, ^{ for (NSInteger i = 0; i < 2; i++) { NSLog(@" %@ ======= %ld", [NSThread currentThread], i); } }); //async 异步函数,会等当前函数执行完后在开线程 NSLog(@"asyncConcurrent"); }
运行结果(开启多个线程并行执行)
2016-04-09 14:47:17.777 GCD的使用[2533:62553] {number = 1, name = main}2016-04-09 14:47:17.778 GCD的使用[2533:62553] asyncConcurrent
2016-04-09 14:47:17.778 GCD的使用[2533:62605] {number = 3, name = (null)} ======= 0
2016-04-09 14:47:17.778 GCD的使用[2533:62606] {number = 2, name = (null)} ======= 0
2016-04-09 14:47:17.778 GCD的使用[2533:62607] {number = 4, name = (null)} ======= 0
2016-04-09 14:47:17.779 GCD的使用[2533:62605] {number = 3, name = (null)} ======= 1
2016-04-09 14:47:17.779 GCD的使用[2533:62606] {number = 2, name = (null)} ======= 1
2016-04-09 14:47:17.779 GCD的使用[2533:62607] {number = 4, name = (null)} ======= 1
使用同步方式创建任务,将任务放入串行队列中
- (void)syncSerial{ //自己创建并发队列 dispatch_queue_t queue = dispatch_queue_create("lcs", DISPATCH_QUEUE_SERIAL); NSLog(@" %@", [NSThread currentThread]); dispatch_sync(queue, ^{ for (NSInteger i = 0; i < 2; i++) { NSLog(@" %@ ======= %ld", [NSThread currentThread], i); } }); dispatch_sync(queue, ^{ for (NSInteger i = 0; i < 2; i++) { NSLog(@" %@ ======= %ld", [NSThread currentThread], i); } }); dispatch_sync(queue, ^{ for (NSInteger i = 0; i < 2; i++) { NSLog(@" %@ ======= %ld", [NSThread currentThread], i); } }); //sync 同步函数,会立刻加入到队列中执行 NSLog(@"asyncConcurrent"); }
运行结果(在当前线程串行执行)
2016-04-09 14:49:48.982 GCD的使用[2577:64118] {number = 1, name = main}2016-04-09 14:49:48.983 GCD的使用[2577:64118] {number = 1, name = main} ======= 0
2016-04-09 14:49:48.983 GCD的使用[2577:64118] {number = 1, name = main} ======= 1
2016-04-09 14:49:48.983 GCD的使用[2577:64118] {number = 1, name = main} ======= 0
2016-04-09 14:49:48.983 GCD的使用[2577:64118] {number = 1, name = main} ======= 1
2016-04-09 14:49:48.983 GCD的使用[2577:64118] {number = 1, name = main} ======= 0
2016-04-09 14:49:48.984 GCD的使用[2577:64118] {number = 1, name = main} ======= 1
2016-04-09 14:49:48.984 GCD的使用[2577:64118] asyncConcurrent
使用同步方式创建任务,将任务放入并行队列中
- (void)syncConcurrent{ //全局并发队列 dispatch_queue_t globalQueue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0); NSLog(@" %@", [NSThread currentThread]); dispatch_sync(globalQueue, ^{ for (NSInteger i = 0; i < 2; i++) { NSLog(@" %@ ======= %ld", [NSThread currentThread], i); } }); dispatch_sync(globalQueue, ^{ for (NSInteger i = 0; i < 2; i++) { NSLog(@" %@ ======= %ld", [NSThread currentThread], i); } }); dispatch_sync(globalQueue, ^{ for (NSInteger i = 0; i < 2; i++) { NSLog(@" %@ ======= %ld", [NSThread currentThread], i); } }); //sync 同步函数,会立刻加入到队列中执行 NSLog(@"asyncConcurrent"); }
运行结果同上一种情况
使用同步方式创建任务,将任务放入主队列中
- (void)syncMainQueue{ //主队列 (加入到主线程中的任务,都在主线程执行) dispatch_queue_t mainQueue = dispatch_get_main_queue(); NSLog(@" %@", [NSThread currentThread]); dispatch_sync(mainQueue, ^{ for (NSInteger i = 0; i < 2; i++) { NSLog(@" %@ ======= %ld", [NSThread currentThread], i); } }); dispatch_sync(mainQueue, ^{ for (NSInteger i = 0; i < 2; i++) { NSLog(@" %@ ======= %ld", [NSThread currentThread], i); } }); dispatch_sync(mainQueue, ^{ for (NSInteger i = 0; i < 2; i++) { NSLog(@" %@ ======= %ld", [NSThread currentThread], i); } }); }
运行结果(任务加入主队列造成互相等待,导致死锁)
2016-04-09 14:49:48.982 GCD的使用[2577:64118] {number = 1, name = main}GCD多线程总结
方案 | 并发队列 | 手动创建串行队列 | 主队列 |
---|---|---|---|
同步(sync) | 没有开启新线程 串行执行 | 没有开启新线程 串行执行 | 没有开启新线程 串行执行 |
异步(async) | 开启新线程 并行执行 | 开启新线程 串行执行 | 没有开启新线程 串行执行 |
NSOperation的使用方法
NSOperation的使用与GCD类似,它是个抽象类,必须使用它的子类,实现多线程操作NSInvocationOperation
NSBlockOperation
自定义类继承自NSOperation,并实现main方法
队列的类型
主队列(凡是添加到主队列中的任务都会放到主线程中执行)手动创建队列(设置队列的最大并发数(maxConcurrentOperationCount)为1,则该队列为串行队列)
创建任务
- (void)viewDidLoad { [super viewDidLoad]; //创建队列 _queue = [[NSOperationQueue alloc] init]; //设置最大并发数,设置为1,则该队列就是串行队列 _queue.maxConcurrentOperationCount = 3; } - (void)createOperation{ //创建任务 //方式一 NSInvocationOperation *op1 = [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(run) object:nil]; //方式二 NSBlockOperation *op2 = [NSBlockOperation blockOperationWithBlock:^{ NSLog(@"messionBlock %@", [NSThread currentThread]); }]; //添加额外的block [op2 addExecutionBlock:^{ NSLog(@"messionExecution %@", [NSThread currentThread]); }]; //方式三:自定义:(实现类的main方法) CSOperation *op3 = [[CSOperation alloc] init]; //加入队列 [_queue addOperation:op1]; //内部调用 [op1 start] [_queue addOperation:op2]; [_queue addOperation:op3]; [_queue addOperationWithBlock:^{ NSLog(@"messionOperationWithBlock %@", [NSThread currentThread]); }]; } - (void)run{ for(int i = 0;i < 10; i++){ NSLog(@"%d messionInvocation %@", i, [NSThread currentThread]); } }
运行结果(开启多个线程,并行执行)
2016-04-16 17:40:23.640 NSOperation的使用[5589:149538] messionExecution {number = 5, name = (null)}2016-04-16 17:40:23.640 NSOperation的使用[5589:149533] messionInvocation {number = 2, name = (null)}
2016-04-16 17:40:23.640 NSOperation的使用[5589:149532] messionmain {number = 3, name = (null)}
2016-04-16 17:40:23.642 NSOperation的使用[5589:149538] messionOperationWithBlock {number = 5, name = (null)}
2016-04-16 17:40:23.640 NSOperation的使用[5589:149531] messionBlock {number = 4, name = (null)}
设置任务之间的依赖,保证执行顺序(依赖设置可以跨队列操作)
- (void)addDependency{ NSBlockOperation *b1 = [NSBlockOperation blockOperationWithBlock:^{ NSLog(@"b1"); }]; NSBlockOperation *b2 = [NSBlockOperation blockOperationWithBlock:^{ for(int i = 0; i < 2; i++){ NSLog(@"b2"); } }]; NSBlockOperation *b3 = [NSBlockOperation blockOperationWithBlock:^{ NSLog(@"b3"); }]; //监听线程执行完毕 b3.completionBlock = ^{ NSLog(@"b3 执行完毕 ---%@", [NSThread currentThread]); }; //b3依赖于b1 b2 [b3 addDependency:b1]; [b3 addDependency:b2]; [_queue addOperation:b1]; [_queue addOperation:b2]; [_queue addOperation:b3]; }
运行结果(b3等待b1,b2任务执行完后再执行)
2016-04-16 17:41:26.994 NSOperation的使用[5618:150401] b22016-04-16 17:41:26.994 NSOperation的使用[5618:150404] b1
2016-04-16 17:41:26.995 NSOperation的使用[5618:150401] b2
2016-04-16 17:41:26.996 NSOperation的使用[5618:150404] b3
2016-04-16 17:41:26.997 NSOperation的使用[5618:150408] b3 执行完毕 —{number = 2, name = (null)}
相关文章推荐
- iOS中倒计时
- iOS开发-由浅至深学习block
- iOS——Quartz2D
- iOS学习笔记30-系统服务(三)蓝牙
- iOS学习笔记30-系统服务(三)蓝牙
- 离屏 渲染
- iOS 基本编码格式转化
- autolayout 设置使用tableviewHeaderView
- iOSapp的json告示
- IOS工具方法小节
- iOS顶部 图片 文字 button (底部横线选中) 切换
- iOS蓝牙中的进制转换,数据格式转换
- iOS中点击button切换背景图片与点击切换背景颜色问题的解释
- iOS的URL处理
- iOS 国际化 检测格式是否正确
- iOS9 之后设置状态栏的颜色
- iOS 动画Animation-4-5: CALayer子类:CATransformLayer
- ios 保存和加载iphone document目录里面的图片
- 【iOS】手动抛出异常
- iOS开发中内存警告