基本概念
NSOperation的作用
配合使用NSOperation和NSOperationQueue也能实现多线程编程
NSOperation和NSOperationQueue实现多线程的具体步骤:
先将需要执行的操作封装到一个NSOperation对象中
然后将NSOperation对象添加到NSOperationQueue中
系统会自动将NSOperationQueue中的NSOperation取出来
将取出的NSOperation封装的操作放到一条新线程中执行
NSOperation用来封装操作
NSOperation是个抽象类,并不具备封装操作的能力,必须使用它的子类
使用NSOperation子类的方式有3种
NSInvocationOperation
NSBlockOperation
自定义子类继承NSOperation,实现内部相应的方法
NSOperationQueue用来存放任务
实现步骤
①封装操作
②把操作添加到队列
NSOperation的基本使用
NSInvocationOperation
创建NSInvocationOperation对象
-(id)initWithTarget:(id)target selector:(SEL)sel object:(id)arg;
调用start方法开始执行操作
-(void)start;
一旦执行操作,就会调用target的sel方法
注意:默认情况下,调用了start方法后并不会开一条新线程去执行操作,而是在当前线程同步执行操作
只有将NSOperation放到一个NSOperationQueue中,才会异步执行操作
NSBlockOperation
创建NSBlockOperation对象
+(id)blockOperationWithBlock:(void(^)(void))block;
通过addExecutionBlock:方法添加更多的操作
-(void)addExecutionBlock:(void(^)(void))block;
注意:只要NSBlockOperation封装的操作数 > 1,就会异步执行操作
①封装操作
②追加任务
③追加的任务在子线程中执行
自定义NSOperation
自定义NSOperation的步骤很简单
重写-(void)main方法,在里面实现想执行的任务
重写-(void)main方法的注意点
自己创建自动释放池(因为如果是异步操作,无法访问主线程的自动释放池)
经常通过-(BOOL)isCancelled方法检测操作是否被取消,对取消做出响应
实现:重写内部的Main方法,说明start方法内部也是调用的main方法
优点:有利于代码的封装和复用
//1.封装操作/*
第一个参数:目标对象
第二个参数:该操作要调用的方法,最多接受一个参数
第三个参数:调用方法传递的参数,如果方法不接受参数,那么该值传nil
*/NSInvocationOperation *operation = [[NSInvocationOperation alloc] initWithTarget:selfselector:@selector(run) object:nil];
//2.启动操作[operation start];
-------------------------------------------------
// 02 NSBlockOperation
//1.封装操作/*
NSBlockOperation提供了一个类方法,在该类方法中封装操作
*/NSBlockOperation *operation = [NSBlockOperation blockOperationWithBlock:^{
//在主线程中执行
NSLog(@"---download1--%@",[NSThreadcurrentThread]);
}];
//2.追加操作,追加的操作在子线程中执行
[operation addExecutionBlock:^{
NSLog(@"---download2--%@",[NSThreadcurrentThread]);
}];
[operation addExecutionBlock:^{
NSLog(@"---download3--%@",[NSThreadcurrentThread]);
}];
//3.启动执行操作[operation start];
// 03 自定义NSOperation//如何封装操作?
//自定义的NSOperation,通过重写内部的main方法实现封装操作
-(void)main {
NSLog(@"--main--%@",[NSThreadcurrentThread]);
}//如何使用?
//1.实例化一个自定义操作对象
XMGOperation *op = [[XMGOperation alloc]init];
//2.执行操作
[op start];
NSOperationQueue的基本使用
lNSOperationQueue的作用
NSOperation可以调用start方法来执行任务,但默认是同步执行的
如果将NSOperation添加到NSOperationQueue(操作队列)中,系统会自动异步执行NSOperation中的操作
添加操作到NSOperationQueue中
-(void)addOperation:(NSOperation*)op;
-(void)addOperationWithBlock:(void(^)(void))block;
队列
回顾GCD中的四种队列
说明主队列和非主队列的关系和区别:
1.凡是添加到主队列中的任务都在主线程中执行
2.非主队列同时具备了串行和并发的功能,默认是并发队列
基本使用
NSInvocationOperation+非主队列
NSBlockOperation+非主队列
①封装任务
②追加任务
③简便方法
自定义NSOperation+非主队列
注意:把操作添加到队列中的方法内部会自动调用start方法
其它使用
①最大并发数
作用:通过设置最大并发数可以控制队列是并发的还是串行的,如果大于1那么队列里面的任务并发执行
maxConcurrentOperationCount==3//同一时间最多只能执行三个任务,当大于1的时候并发执行
maxConcurrentOperationCount==1//队列里面的所有任务串行执行
默认值:默认值为-1,默认是一个并发队列
最大并发数的相关方法
-(NSInteger)maxConcurrentOperationCount;
-(void)setMaxConcurrentOperationCount:(NSInteger)cnt;
②队列的挂起
暂停和恢复队列
-(void)setSuspended:(BOOL)b;// YES代表暂停队列,NO代表恢复队列
-(BOOL)isSuspended;
1.当为YES的时候表示暂停,当为NO的时候表示继续
2.任务是有状态的,如等待执行,正在执行等,挂起操作只能挂起当前非处于执行状态的任务
③队列的取消
-(void)cancelAllOperations;
提示:也可以调用NSOperation的-(void)cancel方法取消单个操作
1.内部调用了每个任务的取消方法
2.取消是不可恢复的
3.如果自定义NSOperation苹果官方的建议
④操作依赖和监听
1.同一个队列中的任务的依赖
[operationB addDependency:operationA];// 操作B依赖于操作A
2.注意不要循环依赖,循环依赖的任务将不会被执行
3.跨队列依赖:可以在不同queue的NSOperation之间创建依赖关系
4.completionBlock
可以监听一个操作的执行完毕
-(void(^)(void))completionBlock;
-(void)setCompletionBlock:(void(^)(void))block;
线程间通信
1.开子线程下载图片回到主线程刷新UI
(1)开子线程下载图片
//1.创建队列NSOperationQueue *queue = [[NSOperationQueue alloc]init];
//2.使用简便方法封装操作并添加到队列中
[queue addOperationWithBlock:^{
//3.在该block中下载图片
NSURL*url =[NSURLURLWithString:@"http://news.51sheyuan.com/uploads/allimg/111001/133442IB-2.jpg"];
NSData*data = [NSDatadataWithContentsOfURL:url];
UIImage*image = [UIImageimageWithData:data];
NSLog(@"下载图片操作--%@",[NSThreadcurrentThread]);
//4.回到主线程刷新
UI[[NSOperationQueue mainQueue] addOperationWithBlock:^{
self.imageView.image= image;
NSLog(@"刷新UI操作---%@",[NSThreadcurrentThread]);
}];
}];
2.下载并合成图片最后回到主线程刷新UI
- (void)download2 {
NSLog(@"----");
//1.创建队列
NSOperationQueue *queue = [[NSOperationQueue alloc]init];
//2.封装操作下载图片1
NSBlockOperation *op1 = [NSBlockOperation blockOperationWithBlock:^{
NSURL *url = [NSURL URLWithString:@"http://h.hiphotos.baidu.com/zhidao/pic/item/6a63f6246b600c3320b14bb3184c510fd8f9a185.jpg"];
NSData *data = [NSData dataWithContentsOfURL:url];
//拿到图片数据 self.image1 = [UIImage imageWithData:data];
}];
//3.封装操作下载图片2
NSBlockOperation *op2 = [NSBlockOperation blockOperationWithBlock:^{
NSURL *url = [NSURL URLWithString:@"http://pic.58pic.com/58pic/13/87/82/27Q58PICYje_1024.jpg"]; NSData *data = [NSData dataWithContentsOfURL:url];
//拿到图片数据
self.image2 = [UIImage imageWithData:data]; }];
//4.合成图片
NSBlockOperation *combine = [NSBlockOperation blockOperationWithBlock:^{
//4.1 开启图形上下文
UIGraphicsBeginImageContext(CGSizeMake(200, 200));
//4.2 画image1
[self.image1 drawInRect:CGRectMake(0, 0, 200, 100)];
//4.3 画image2
[self.image2 drawInRect:CGRectMake(0, 100, 200, 100)];
//4.4 根据图形上下文拿到图片数据
UIImage *image = UIGraphicsGetImageFromCurrentImageContext();
// NSLog(@"%@",image);
//4.5 关闭图形上下文
UIGraphicsEndImageContext();
//7.回到主线程刷新UI
[[NSOperationQueue mainQueue]addOperationWithBlock:^{
self.imageView.image = image;
NSLog(@"刷新UI---%@",[NSThread currentThread]);
}];
}];
//5.设置操作依赖
[combine addDependency:op1];
[combine addDependency:op2];
//6.添加操作到队列中执行
[queue addOperation:op1];
[queue addOperation:op2];
[queue addOperation:combine];
}
多图下载综合案例
一、完成基本数据的展示
问题1:在子线程中下载图片,UI不流畅
问题2:重复下载图片
二、解决重复下载问题
1.对图片进行内存缓存以解决重复下载问题
2.对图片做沙盒缓存,进行离线缓存
3.分析沙盒缓存应该存放到哪个文件夹下面
(1)Documents
该目录下面的数据在连接手机时会备份
苹果官方不允许把下载的数据存放于该目录下
(2)Library
1.caches
2.perference:该目录用来存放偏好设置如登录名密码等等
(3)Tmp:会被随机删除
三、解决UI不流畅问题
1.解决思路:把下载图片的操作放在子线程中处理
2.新的问题:
①图片不显示:没有刷新
②线程间的通信:在子线程下载图片写入沙盒等,回到主线程刷新UI
③重复下载问题:需要对操作进行缓存
④数据错乱问题:cell的复用造成,通过设置占位图片解决该问题
3.注意点
(1)图片下载完成之后把操作从缓存中移除
(2)图片下载完成后,做容错处理
(3)发生内存警告如何处理