iOS 之路iOS技术专题

NSOperation

2016-05-15  本文已影响479人  fanzq_jianshu

NSOperation

NSOperation与GCD队列+操作的差别:

    01. 同步与异步函数:
        1.1 同步函数:NSOperation中,任务的start方法是同步函数,start是让任务在当前线程中同步串行执行,并且调用内部的main方法
            a  如果要让任务在当前线程执行,就直接创建任务并start,而不是加入到队列中

        1.2 异步函数:NSOperationQueue都是异步函数。加入到NSOperationQueue中的任务,内部会调用start方法,让任务在当前线程中同步串行执行,并且调用内部的main方法

        1.3 NSBlockOperation追加任务也是在新开线程执行,是特例。

    02 NSOperation的队列:
        1. 主队列的任务在主线程中执行(同GCD);
        2. 非主队列的任务同时具备串行和并发的能力,通过最大并发数设置队列模式(串行?并发)。



1.NSOperation和GCD的对比?
1) GCD是纯C语言的API,而操作队列则是Object-C的对象。
2)在GCD中,任务用块(block)来表示,而块是个轻量级的数据结构;
   相反操作队列中的『操作』NSOperation则是个更加重量级的Object-C对象。

操作队列的好处有:
    1)NSOperationQueue可以方便的调用cancel方法来取消某个操作,而GCD中的任务是无法被取消的(安排好任务之后就不管了)。
    2)NSOperation可以方便的指定操作间的依赖关系。
    3)NSOperation可以通过KVO提供对NSOperation对象的精细控制。
    4)NSOperation可以方便的指定操作优先级。
    5)通过自定义NSOperation的子类可以实现操作重用。

NSOperation的使用

(1)基本使用步骤

01 NSOperation本身是抽象类,只有它的子类才具备封装操作的能力
02 三个子类分别是:NSBlockOperation、NSInvocationOperation以及自定义继承自NSOperation的类
03 NSOperation和NSOperationQueue结合使用实现多线程并发具体步骤:
    3.1 先将需要执行的操作封装到一个NSOperation对象中
    3.2 然后将NSOperation对象添加到NSOperationQueue中
    3.3 系统会自动将NSOperationQueue中的NSOperation取出来
    3.4 将取出的NSOperation封装的操作放到新的线程中执行(开多少线程由系统管理)

(2)基本使用代码

  01 NSInvocationOperation
    //1.封装操作
    /*
     第一个参数:目标对象
     第二个参数:该操作要调用的方法,最多接受一个参数
     第三个参数:调用方法传递的参数,如果方法不接受参数,那么该值传nil
     */
    NSInvocationOperation *operation = [[NSInvocationOperation alloc]
                                        initWithTarget:self selector:@selector(run) object:nil];

    //2.启动操作
    [operation start];//开始调用run方法

-------------------------------------------------
  02 NSBlockOperation
    //1.封装操作
    NSBlockOperation *operation = [NSBlockOperation blockOperationWithBlock:^{

        //在主线程中执行
        NSLog(@"---download1--%@",[NSThread currentThread]);
    }];

    //2.追加操作,追加的操作在子线程中执行
    [operation addExecutionBlock:^{
        NSLog(@"---download2--%@",[NSThread currentThread]);
    }];

    //3.启动执行操作
    [operation start];

----------------------------------------------
 03 自定义NSOperation
    //如何封装操作?
    //自定义的NSOperation,通过重写内部的main方法实现封装操作
    //重写- (void)main方法的注意点
自己创建自动释放池(因为如果是异步操作,无法访问主线程的自动释放池)
    //经常通过- (BOOL)isCancelled方法检测操作是否被取消,对取消做出响应

//优点:
//1.代码简洁
//2.提高代码的复用性
//3.信息隐蔽
自定义FZQOperation.m中
    -(void)main
    {
        NSLog(@"--main--%@",[NSThread currentThread]);
    }

    //如何使用?
    //1.实例化一个自定义操作对象
    XMGOperation *op = [[XMGOperation alloc]init];

    //2.执行操作
    [op start];
    //start方法内部会调用main方法,所以可以把线程调用的代码放在内部的main方法中

NSOperationQueue基本使用

(1)NSOperation中的两种队列

01 主队列 通过mainQueue获得,凡是放到主队列中的任务都将在主线程执行
02 非主队列 直接alloc init出来的队列。非主队列同时具备了并发和串行的功能,通过设置最大并发数属性来控制任务是并发执行还是串行执行。具备新开线程能力

(2)相关代码

    01 自定义NSOperation
-(void)customOperation
{
    //1.创建队列
    NSOperationQueue *queue = [[NSOperationQueue alloc]init];

    //2.封装操作
    //好处:1.信息隐蔽
    //2.代码复用

    XMGOperation *op1 = [[XMGOperation alloc]init];
    XMGOperation *op2 = [[XMGOperation alloc]init];

    //3.添加操作到队列中
    [queue addOperation:op1];
    [queue addOperation:op2];
}

    02 NSBlockOperation
- (void)block
{
    //1.创建队列
    NSOperationQueue *queue = [[NSOperationQueue alloc]init];

    //2.封装操作
    NSBlockOperation *op1 = [NSBlockOperation blockOperationWithBlock:^{
        NSLog(@"1----%@",[NSThread currentThread]);
    }];

    NSBlockOperation *op2 = [NSBlockOperation blockOperationWithBlock:^{
        NSLog(@"2----%@",[NSThread currentThread]);

    }];

    [op2 addExecutionBlock:^{
        NSLog(@"3----%@",[NSThread currentThread]);
    }];

    [op2 addExecutionBlock:^{
        NSLog(@"4----%@",[NSThread currentThread]);
    }];

    //3.添加操作到队列中
    [queue addOperation:op1];
    [queue addOperation:op2];

    //补充:简便方法
    [queue addOperationWithBlock:^{
        NSLog(@"5----%@",[NSThread currentThread]);
    }];

}

    03 NSInvocationOperation
- (void)invocation
{
    /*
     GCD中的队列:
     串行队列:自己创建的,主队列
     并发队列:自己创建的,全局并发队列

     NSOperationQueue
     主队列:[NSOperationQueue mainqueue];凡是放在主队列中的操作都在主线程中串行执行
     非主队列:[[NSOperationQueue alloc]init],并发和串行,默认是并发执行的
     */

    //1.创建队列
    NSOperationQueue *queue = [[NSOperationQueue alloc]init];

    //2.封装操作
    NSInvocationOperation *op1 = [[NSInvocationOperation alloc]initWithTarget:self selector:@selector(download1) object:nil];

    NSInvocationOperation *op2 = [[NSInvocationOperation alloc]initWithTarget:self selector:@selector(download2) object:nil];

    //3.把封装好的操作添加到队列中
    [queue addOperation:op1];
    [queue addOperation:op2];
}

NSOperation其它用法

(1)设置最大并发数【控制任务并发和串行】

//1.创建队列
    NSOperationQueue *queue = [[NSOperationQueue alloc]init];

    //2.设置最大并发数
    //注意点:该属性需要在任务添加到队列中之前进行设置
    //该属性控制队列是串行执行还是并发执行
    //如果最大并发数等于1,那么该队列是串行的,如果大于1那么是并行的
    //系统的最大并发数有个默认的值,为-1,如果该属性设置为0,那么不会执行任何任务
    queue.maxConcurrentOperationCount = 2;

(2)暂停和恢复以及取消

    //设置暂停和恢复
    //suspended设置为YES表示暂停,suspended设置为NO表示恢复
    //暂停表示不继续执行队列中的下一个任务,暂停操作是可以恢复的
    self.queue.suspended = !self.queue.suspended;


    //取消队列里面的所有操作
    //取消之后,当前正在执行的操作的下一个操作将不再执行,而且永远都不在执行,就像后面的所有任务都从队列里面移除了一样
    //取消操作是不可以恢复的
    [self.queue cancelAllOperations];

---------自定义NSOperation取消操作-----------------------

-(void)main
{
    //耗时操作1
    for (int i = 0; i<1000; i++) {
        NSLog(@"任务1-%d--%@",i,[NSThread currentThread]);
    }
    NSLog(@"+++++++++++++++++++++++++++++++++");

    //苹果官方建议,每当执行完一次耗时操作之后,就查看一下当前队列是否为取消状态,如果是,那么就直接退出
    //好处是可以提高程序的性能
    if (self.isCancelled) {
        return;
    }

    //耗时操作2
    for (int i = 0; i<1000; i++) {
        NSLog(@"任务1-%d--%@",i,[NSThread currentThread]);
    }

    NSLog(@"+++++++++++++++++++++++++++++++++");
}

(3)操作依赖

01 NSOperation之间可以设置依赖来保证执行顺序
- (void)addOperation:(NSOperation *)op;
- (void)addOperationWithBlock:(void (^)(void))block;

//添加操作依赖,B在A之后执行
[operationB addDependency:operationA];

02 可以在不同queue的NSOperation之间创建依赖关系

03注意:不能相互依赖
比如A依赖B,B依赖A,否则AB及依赖AB的操作都不会执行(不会崩溃)
    /*
    [download addDependency:op4];
    [op4 addDependency:download];
    */

(4)操作的监听

可以监听一个操作的执行完毕
- (void (^)(void))completionBlock;
- (void)setCompletionBlock:(void (^)(void))block;

(1)开子线程下载图片

 //1.创建队列
    NSOperationQueue *queue = [[NSOperationQueue alloc]init];

    //2.使用简便方法封装操作并添加到队列中
    [queue addOperationWithBlock:^{

        //3.在该block中下载图片
        NSURL *url = [NSURL URLWithString:@"http://news.51sheyuan.com/uploads/allimg/111001/133442IB-2.jpg"];
        NSData *data = [NSData dataWithContentsOfURL:url];
        UIImage *image = [UIImage imageWithData:data];
        NSLog(@"下载图片操作--%@",[NSThread currentThread]);

        //4.回到主线程刷新UI
        [[NSOperationQueue mainQueue] addOperationWithBlock:^{
            self.imageView.image = image;
            NSLog(@"刷新UI操作---%@",[NSThread currentThread]);
        }];
    }];

(2)下载多张图片合成综合案例(设置操作依赖)

//02 综合案例
- (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];
    }
上一篇 下一篇

猜你喜欢

热点阅读