美文网首页
NSOperation

NSOperation

作者: yuqian__zhang | 来源:发表于2018-01-09 17:04 被阅读0次

    基本概念

    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)发生内存警告如何处理

    相关文章

      网友评论

          本文标题:NSOperation

          本文链接:https://www.haomeiwen.com/subject/opcgnxtx.html