iOS开发 - 多线程实现方案之NSOperation篇
NSOperation简介
1.实现多线程编程步骤:
配合使用NSOperation和NSOperationQueue实现多线程编程,我们不用考虑线程的生命周期、同步、加锁等问题,如下:
- 先将需要执行的操作封装到一个NSOperation对象中
- 然后将NSOperation对象添加到NSOperationQueue中
- 系统会自动将NSOperation中封装的操作放到一条新线程中执行
2.NSOperation的子类:
NSOperation是个抽象类,并不具备封装操作的能力,必须使⽤它的子类,使用NSOperation⼦类的方式有3种:
- NSInvocationOperation
- NSBlockOperation
- 自定义子类继承NSOperation,实现内部相应的⽅法
封装操作到NSOperation对象中(步骤一)
1.NSInvocationOperation
- (void)invocationOperation {
//封装操作(任务)
//注意:默认情况下,调用了start方法后并不会开一条新线程去执行操作,而是在当前线程同步执行操作。只有将operation放到一个NSOperationQueue中,才会异步执行操作
NSInvocationOperation *op1 = [[NSInvocationOperation alloc]initWithTarget:self selector:@selector(startExecute:) object:@"funky"];
[op1 start];
}
- (void)startExecute:(NSString *)param {
NSLog(@"-------线程:%@ ------ 参数:%@",[NSThread currentThread],param);
//执行操作...
}
NSInvocationOperation封装操作
2.NSBlockOperation
- (void)blockOperation {
//注意:如果一个操作中的任务数量大于1,那么会开子线程并发执行任务,但不一定是子线程,有可能是主线程
NSBlockOperation *op = [NSBlockOperation blockOperationWithBlock:^{
NSLog(@"----- %@",[NSThread currentThread]);
//执行操作...
}];
NSBlockOperation *op1 = [NSBlockOperation blockOperationWithBlock:^{
NSLog(@"----- %@",[NSThread currentThread]);
//执行操作...
}];
[op1 addExecutionBlock:^{
//op1追加任务1...
}];
[op1 addExecutionBlock:^{
//op1追加任务2...
}];
[op start];
[op1 start];
}
NSBlockOperation封装及追加任务
添加到NSOperationQueue中(步骤二)
1.NSInvocationOperation对象添加到NSOperationQueue中
- (void)invocationOperationWithQueue {
//1.封装操作(任务)
NSInvocationOperation *op1 = [[NSInvocationOperation alloc]initWithTarget:self selector:@selector(startExecute:) object:@"funky"];
//2.创建队列(主队列,非主队列)
//主队列 : [NSOperationQueue mainQueue]; (同GCD的主队列,串行队列)
//非主队列 : [[NSOperationQueue alloc]init]; (同时具备并发和串行的功能,默认是并发队列)
NSOperationQueue *queue = [[NSOperationQueue alloc]init];
//3.把操作添加到队列中 (内部已经调用了 [op1 start])
[queue addOperation:op1];
}
- (void)startExecute:(NSString *)param {
NSLog(@"-------线程:%@ ------ 参数:%@",[NSThread currentThread],param);
//执行操作...
}
2.NSBlockOperation对象添加到NSOperationQueue中及简便方式
- (void)blockOperationWithQueue {
NSBlockOperation *op = [NSBlockOperation blockOperationWithBlock:^{
NSLog(@"----- %@",[NSThread currentThread]);
//执行操作...
}];
NSBlockOperation *op1 = [NSBlockOperation blockOperationWithBlock:^{
NSLog(@"----- %@",[NSThread currentThread]);
//执行操作...
}];
//创建队列
NSOperationQueue *queue = [[NSOperationQueue alloc]init];
//添加到队列中
[queue addOperation:op];
[queue addOperation:op1];
}
//或简便方法
-(void)simpleWay{
//创建队列
NSOperationQueue *queue = [[NSOperationQueue alloc]init];
[queue addOperationWithBlock:^{
NSLog(@"开子线程执行任务----%@",[NSThread currentThread]);
}];
}
3.自定义子类继承NSOperation
#import "CustomOperation.h" @implementation CustomOperation //重写main方法,在main方法中 执行操作
//可以提高代码复用性
-(void)main { NSLog(@"----- %@",[NSThread mainThread]); } @end
继承于NSOperation的自定义类.m
- (void)customOperationWithQueue {
CustomOperation *op = [[CustomOperation alloc]init];
NSOperationQueue *queue = [[NSOperationQueue alloc]init];
[queue addOperation:op];
}
使用CustomOperation
NSOperation的其他用法
1.NSOperation系统的子类的取消,暂定操作
//开始执行任务
- (IBAction)startBtnClick:(UIButton *)sender { self.queue = [[NSOperationQueue alloc]init]; //maxConcurrentOperationCount 最大并发数(同一时间最多有多少条任务可以执行)
// >1并发队列 =1串行队列 =0不执行任务 =-1默认,代表最大值,最大并发数不受限制
self.queue.maxConcurrentOperationCount = ; NSBlockOperation *op1 = [NSBlockOperation blockOperationWithBlock:^{ for (int i =; i < ; i++) {
NSLog(@"download1-----%@",[NSThread currentThread]);
} }]; NSBlockOperation *op2 = [NSBlockOperation blockOperationWithBlock:^{ for (int i =; i < ; i++) {
NSLog(@"download2-----%@",[NSThread currentThread]);
} }]; NSBlockOperation *op3 = [NSBlockOperation blockOperationWithBlock:^{ for (int i =; i < ; i++) {
NSLog(@"download3-----%@",[NSThread currentThread]);
} }]; [self.queue addOperation:op1];
[self.queue addOperation:op2];
[self.queue addOperation:op3]; }
//暂停执行任务
- (IBAction)stopBtnClick:(UIButton *)sender { //可恢复,不能暂停当前正处于执行状态的任务
[self.queue setSuspended:YES];
}
//继续执行任务
- (IBAction)goOnBtnClick:(UIButton *)sender { [self.queue setSuspended:NO];
}
//取消执行任务
- (IBAction)cancelBtnClick:(UIButton *)sender { //不可恢复
//该方法内部调用了所有operation的cancel方法
[self.queue cancelAllOperations];
}
2.自定义NSOperation的取消操作
//开始执行任务
- (IBAction)startBtnClick:(UIButton *)sender { self.queue = [[NSOperationQueue alloc]init]; self.queue.maxConcurrentOperationCount = ; HSOperation *op = [[HSOperation alloc]init]; [self.queue addOperation:op]; }
//取消执行任务
- (IBAction)cancelBtnClick:(UIButton *)sender { //不可恢复
//该方法内部调用了所有operation的cancel方法
[self.queue cancelAllOperations];
}
#import "HSOperation.h"
@implementation HSOperation
-(void)main {
//3个耗时操作,但相当于一个任务
for (int i =; i < ; i++) {
//if (self.isCancelled) return; 写在这会耗费性能,不建议在这进行操作
NSLog(@"download1-----%@",[NSThread currentThread]);
}
//检测是否取消了操作
if (self.isCancelled) return;
for (int i =; i < ; i++) {
NSLog(@"download2-----%@",[NSThread currentThread]);
}
if (self.isCancelled) return;
for (int i =; i < ; i++) {
NSLog(@"download3-----%@",[NSThread currentThread]);
}
}
@end
HSOperation.m
3.操作依赖及操作监听
-(void)addDependency{
NSOperationQueue *queue = [[NSOperationQueue alloc]init];
NSBlockOperation *op1 = [NSBlockOperation blockOperationWithBlock:^{
//耗时操作1...
NSLog(@"download1-----%@",[NSThread currentThread]);
}];
NSBlockOperation *op2 = [NSBlockOperation blockOperationWithBlock:^{
//耗时操作2...
NSLog(@"download2-----%@",[NSThread currentThread]);
}];
NSBlockOperation *op3 = [NSBlockOperation blockOperationWithBlock:^{
//耗时操作3...
NSLog(@"download3-----%@",[NSThread currentThread]);
}];
NSBlockOperation *op4 = [NSBlockOperation blockOperationWithBlock:^{
//耗时操作4...
NSLog(@"download4-----%@",[NSThread currentThread]);
}];
//操作监听
op3.completionBlock = ^{
NSLog(@"download3下载完成了-----%@",[NSThread currentThread]);
};
//操作依赖
[op1 addDependency:op2]; //op1 等待 op2执行后 在执行
[op2 addDependency:op4];
[op4 addDependency:op3];
// [op3 addDependency:op1]; 不能循环依赖
[queue addOperation:op1];
[queue addOperation:op2];
[queue addOperation:op3];
[queue addOperation:op4];
}
NSOperation线程间的通信
-(void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event {
NSOperationQueue *queue = [[NSOperationQueue alloc]init];
NSBlockOperation *op = [NSBlockOperation blockOperationWithBlock:^{
NSURL *url = [NSURL URLWithString:@"https://timgsa.baidu.com/timg?image&quality=80&size=b9999_10000&sec=1490335106967&di=7daac700f5a1425d58131134228ea9bc&imgtype=0&src=http%3A%2F%2Fsh.sinaimg.cn%2Fcr%2F2013%2F1204%2F533155851.jpg"];
NSData *imgData = [NSData dataWithContentsOfURL:url];
UIImage *image = [UIImage imageWithData:imgData];
NSLog(@"download---%@",[NSThread currentThread]);
//回到主线程刷新UI
[[NSOperationQueue mainQueue] addOperationWithBlock:^{
self.imageView.image = image;
NSLog(@"UI---%@",[NSThread currentThread]);
}];
}];
[queue addOperation:op];
}
iOS开发 - 多线程实现方案之NSOperation篇的更多相关文章
- iOS开发 - 多线程实现方案之Pthread篇
pthread基础 pthread是POSIX thread的简写,一套通用的多线程API,适用于Unix.Linux.Windows等系统,跨平台.可移植,使用难度大,C语言框架,线程生命周期由程序 ...
- iOS开发 - 多线程实现方案之GCD篇
GCD概念 GCD为Grand Central Dispatch的缩写,纯c语言编写,是Apple开发的一个多核编程的较新的解决方法.它主要用于优化应用程序以支持多核处理器以及其他对称多处理系统.它是 ...
- iOS开发 - 多线程实现方案之NSThread篇
NSThread API //类方法:创建一个线程 + (void)detachNewThreadWithBlock:(void (^)(void))block API_AVAILABLE(macos ...
- iOS开发多线程--技术方案
pthread 实现多线程操作 代码实现: void * run(void *param) { for (NSInteger i = 0; i < 1000; i++) { ...
- iOS开发多线程篇—NSOperation简单介绍
iOS开发多线程篇—NSOperation简单介绍 一.NSOperation简介 1.简单说明 NSOperation的作⽤:配合使用NSOperation和NSOperationQueue也能实现 ...
- iOS开发多线程篇—NSOperation基本操作
iOS开发多线程篇—NSOperation基本操作 一.并发数 (1)并发数:同时执⾏行的任务数.比如,同时开3个线程执行3个任务,并发数就是3 (2)最大并发数:同一时间最多只能执行的任务的个数. ...
- iOS开发多线程篇—自定义NSOperation
iOS开发多线程篇—自定义NSOperation 一.实现一个简单的tableView显示效果 实现效果展示: 代码示例(使用以前在主控制器中进行业务处理的方式) 1.新建一个项目,让控制器继承自UI ...
- iOS开发多线程篇 09 —NSOperation简单介绍
iOS开发多线程篇—NSOperation简单介绍 一.NSOperation简介 1.简单说明 NSOperation的作⽤:配合使用NSOperation和NSOperationQueue也能实现 ...
- iOS开发多线程篇 11 —自定义NSOperation
iOS开发多线程篇—自定义NSOperation 一.实现一个简单的tableView显示效果 实现效果展示: 代码示例(使用以前在主控制器中进行业务处理的方式) 1.新建一个项目,让控制器继承自UI ...
随机推荐
- Activity和ListActivity的区别
http://book.51cto.com/art/201007/212051.htm
- SpringBoot-(8)-配置MySQL数据库链接,配置数据坚挺拦截,创建默认数据表
一,链接mysql数据库 # 数据源基本配置 spring.datasource.username=root spring.datasource.password=123456 spring.data ...
- [数据挖掘课程笔记]Naïve Bayesian Classifier
朴素贝叶斯模型 1) X:一条未被标记的数据 2) H:一个假设,如H=X属于Ci类 根据贝叶斯公式 把X表示为(x1,x2,....xn) x1,x2,....xn表示X在各个特征上的值. 假设有c ...
- mac10.10 打造Python多版本虚拟环境
一.简介 1.在操作系统mac10.10(yosemite)中搭建Python多版本虚拟环境: 2.多版本虚拟环境包含三个含义: 在一个操作系统中安装多个版本的Python,不同版本可以随意切换,例如 ...
- 编写按键驱动以及在framework层上报按键事件
平台信息:内核:linux3.10 系统:android6.0平台:RK3288 前言:本文主要实现的功能是在android系统中添加一个按键,在驱动层使用定时器,每隔1秒钟向上层发送按键实现,fra ...
- LightOJ1234 Harmonic Number —— 分区打表
题目链接:https://vjudge.net/problem/LightOJ-1234 1234 - Harmonic Number PDF (English) Statistics Foru ...
- 基于官方驱动封装mongodb
还是一如既往先把结构图放出来,上上个版本添加了redis的缓存,但是不满足我的需求,因为公司有项目要求是分布式所以呢,这里我就增加了mongoDb进行缓存分布式,好了先看结构图(1). 总的来说比较蛋 ...
- 小工具之Synergy
用于两个主机共享键盘和鼠标的工具: 软件名字:synergy软件主页: http://synergy-foss.org支持平台:linux,mac,windows 通吃 作用:通过网络在多台主机之间共 ...
- 使用PowerDesigner设计数据库
1.快捷键CTRL+N 创建 New Model 选择如下图,并设置 Model name 单击OK 2.使用工具添加实体 双击Entity_1,填上如下图信息 切换选项卡,添加属性信息 其中 M ...
- js 异步问题
如果发现前端的异步请求,network有发送成功而且有返回值,但是没有进前端的callback成功回掉函数内,这时候就需要检查是不是配置文件做了语法限制,只能返回某些数据格式,不能返回某些数据格式. ...