NSOperation、Queue使用注意点总结

NSOperation、NSOperationQueue 是苹果提供给我们的一套多线程解决方案。实际上 NSOperation、NSOperationQueue 是基于 GCD 更高一层的封装,完全面向对象。但是比 GCD 更简单易用、代码可读性也更高。

1.NSOperation基本使用

NSOperation可以调用start方法来执行任务,但默认是同步执行的
NSOperation是个抽象类,并不具备封装操作的能力,必须使用它的子类(在第三条介绍子类)

  • 先将需要执行的操作封装到一个NSOperation对象中
  • 然后将NSOperation对象添加到NSOperationQueue中
  • 系统会自动将NSOperationQueue中的NSOperation取出来
  • 将取出的NSOperation封装的操作放到一条新线程中执行

2.NSOperationQueue 队列

    //创建操作(任务)
    NSOperationQueue  *queue = [[NSOperationQueue alloc]init];
    
    NSInvocationOperation  *operationInvocation = [[NSInvocationOperation alloc]initWithTarget:self selector:@selector(downLoad1) object:nil];
    NSBlockOperation  *operationblock = [NSBlockOperation blockOperationWithBlock:^{
        NSLog(@"downLoad3-----%@",[NSThread currentThread]);
        
    }];
    //添加任务
    [queue addOperation:operationInvocation];//自动调用 [op start]
    [queue addOperation:operationblock];//自动调用 [op start]
  • 通过 [[NSOperationQueue alloc]init]创建的操作队列,其内添加的任务在子线程执行。
  • 通过[NSOperationQueue mainQueue] 获取的是主操作队列,其内添加的任务(NSOperation)在主线程执行。
  • 如果将NSOpeeration添加到NSOperationQueue(操作队列)中,系统自动异步执行NSOperation中的操作。

添加操作到NSOperationQueue中
//add后的任务不需要调用-start就会自动执行

  • (void)addOperation:(NSOperation *)op;
  • (void)addOperationWithBlock

3.NSOperation的子类

3.1 NSInvocationOperation

//如果是在主线程创建,就在主线程执行任务。如果是子线程,那就在子线程
  NSInvocationOperation  *operation = [[NSInvocationOperation alloc]initWithTarget:self selector:@selector(run) object:nil];
  [operation start];
  • 默认情况下,调用了start方法后并不会开一条新线程去执行操作,而是在当前线程同步执行操作
  • 只有将NSOperation放到一个NSOperationQueue中,才会异步执行操作

3.2NSBlockOperation

    //包装的任务在主线程
    NSBlockOperation  *operation = [NSBlockOperation blockOperationWithBlock:^{
        NSLog(@"1  --- %@",[NSThread currentThread]);
    }];
    //再添加的任务 会在新子线程执行
    [operation addExecutionBlock:^{
        NSLog(@"2 --- %@",[NSThread currentThread]);  
    }];
    [operation start];
  • 只要NSBlockOperation封装的操作数 > 1,就会异步执行操作

3.3自定义NSOperation

  • 自定义NSOperation的步骤很简单,只要重写- (void)main方法,在里面实现想执行的任务。当然如果结构很复杂的自定义NSOperation还需要实现- (BOOL)isConcurrent是否并发等方法。
  • 经常通过- (BOOL)isCancelled方法检测操作是否被取消,对取消做出响应
//需要执行的任务
- (void)main{
    
    for (int i =0; i<10088; i++) {
        NSLog(@"XBSOpertion----下载任务%@",[NSThread currentThread]);
    }
    //这句写与否直接就看出你的水平
    if (self.isCancelled) return;
    
    for (int i =0; i<10088; i++) {
        NSLog(@"XBSOpertion----下载任务%@",[NSThread currentThread]);
    }
}

4.最大并发数maxConcurrentOperationCount

系统默认最大并发数是NSOperationQueueDefaultMaxConcurrentOperationCount = -1.

  • maxConcurrentOperationCount = 0 不执行队列
  • maxConcurrentOperationCount = 1 串行队列
  • maxConcurrentOperationCount > 1 并发队列

5.NSOperation挂起和取消

注意:
1.这里的暂停和取消(包括操作的取消和队列的取消)并不代表可以将当前的操作立即取消,而是当当前的操作执行完毕之后不再执行新的操作。
2.暂停和取消的区别就在于:暂停操作之后还可以恢复操作,继续向下执行;而取消操作之后,所有的操作就清空了,无法再接着执行剩下的操作

5.1挂起suspended.

  • suspended YES:暂停(挂起)队列 NO:恢复队列

5.2 取消

  • 单个任务取消 [operation cancel];
  • 整个操作队列所有未开始任务的取消 [self.queue cancelAllOperations]

6.依赖和监听

6.1依赖[queue addExecution]

6.1.1 同队列内依赖

NSOperation之间可以设置依赖来保证执行顺序
比如一定要让操作A执行完后,才能执行操作B,可以这么写
[operationB addDependency:operationA]; // 操作B依赖于操作A

6.1.2 跨队列依赖

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


跨队列依赖.jpeg

图解:NSOperation3依赖于NSOperation2,NSOperation2又依赖于其他队列的NSOperation6,NSOperation6又依赖于NSOperation1。

只要是个NSOperation对象就可以建立依赖,非常强大。

注意:不能相互依赖。比如A依赖B,B依赖A

6.2 监听

可以监听一个操作的执行完毕

  • (void (^)(void))completionBlock;
  • (void)setCompletionBlock:(void (^)(void))block;

注意点:苹果官方描述:
A finished (or canceled) operation is still given a chance to execute its completion block before it is removed from the queue

一个已经完成(或者已经被取消)的操作在被移除操作队列之前仍然有机会去执行completion block。

7.知识实践

下载两张图片,然后合成一张显示处出啦

- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{
    
    NSOperationQueue  *queueOp = [[NSOperationQueue alloc]init];
    __block UIImage  *image1 = nil;
    
    //下载图片1
    NSBlockOperation  *downLoad1 = [NSBlockOperation blockOperationWithBlock:^{
        NSURL  *url = [NSURL URLWithString:@"https://upload-images.jianshu.io/upload_images/6687791-5ad29d1a5c1c84db.jpeg?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240"];

        image1 = [UIImage imageWithData:[NSData dataWithContentsOfURL:url]];
        
    }];

    __block  UIImage  *image2 = nil;
    NSBlockOperation  *downLoad2 = [NSBlockOperation blockOperationWithBlock:^{
       
        NSURL  *url = [NSURL URLWithString:@"https://upload-images.jianshu.io/upload_images/6687791-2a51b4b27040ed41.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240"];
        
        image2 = [UIImage imageWithData:[NSData dataWithContentsOfURL:url]];
    }];
 
    
    
    //4.等待两张图片下载完成,将两张图片合成一张
    
    NSBlockOperation  *combine = [NSBlockOperation blockOperationWithBlock:^{
        //开启新的图形上下文
        UIGraphicsBeginImageContext(CGSizeMake(300, 300));
        //绘制图片
        [image1 drawInRect:CGRectMake(0, 0, 300, 150)];
        image1 = nil;
        [image2 drawInRect:CGRectMake(0, 150, 300, 150)];
        image2 = nil;
        //取得图形上下文中的图片
        UIImage  *newImage = UIGraphicsGetImageFromCurrentImageContext();
        //结束上下文
        UIGraphicsEndImageContext();
        //回到主线程显示图片
       
        [[NSOperationQueue mainQueue] addOperationWithBlock:^{
            self.imageView.image = newImage;
        }];
        
        
    }];
    //combine合成任务的执行依赖 downLoad1  和 downLoad2的执行完成
    [combine addDependency:downLoad1];
    [combine addDependency:downLoad2];
    
    //三个任务添加到操作队列中
    [queueOp addOperation:downLoad1];
    [queueOp addOperation:downLoad2];
    [queueOp addOperation:combine];
    
}

如果您有什么疑问或者发现书写歧义,非常感激您能留言~

©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

友情链接更多精彩内容