iOS多线程开发之NSOperation用法总结

NSOperation的作用

配合使用NSOperation和NSOperationQueue也能实现多线程编程

NSOperation和NSOperationQueue实现多线程的具体步骤

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

注意:

  • NSOperation是个抽象类,并不具备封装操作的能力,必须使用它的子类
  • 使用NSOperation子类的方式有3种:
  • NSInvocationOperation
  • NSBlockOperation
  • 自定义子类继承NSOperation,实现内部相应的方法

NSInvocationOperation类

  • 创建NSInvocationOperation对象
-(id)initWithTarget:(id)targetselector:(SEL)selobject:(id)arg;
  • 调用start方法开始执行操作
-(void)start;

一旦执行操作,就会调用target的sel方法

  • 代码:
NSInvocationOperation *op = [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(run) object:nil];
    [op start];
 - (void)run
{
    NSLog(@"------%@", [NSThread currentThread]);
}
  • 注意:
  • 默认情况下,调用了start方法后并不会开一条新线程去执行操作,而是在当前线程同步执行操作
  • 只有将NSOperation放到一个NSOperationQueue中,才会异步执行操作
  • 此类仅当了解,在开发中并不常用

NSBlockOperation类

  • 创建NSBlockOperation对象
+(id)blockOperationWithBlock:(void(^)(void))block;
  • 通过addExecutionBlock:方法添加更多的操作
-(void)addExecutionBlock:(void(^)(void))block;
  • 代码:
 NSBlockOperation *op = [NSBlockOperation blockOperationWithBlock:^{
        // 在主线程
        NSLog(@"下载1------%@", [NSThread currentThread]);
    }];
    
    // 添加额外的任务(在子线程执行)
    [op addExecutionBlock:^{
        NSLog(@"下载2------%@", [NSThread currentThread]);
    }];
    [op addExecutionBlock:^{
        NSLog(@"下载3------%@", [NSThread currentThread]);
    }];
    [op addExecutionBlock:^{
        NSLog(@"下载4------%@", [NSThread currentThread]);
    }];
    
    [op start];
  • 注意:只要NSBlockOperation封装的操作数 >1,就会异步执行操作

NSOperationQueue

  • NSOperationQueue的作用

  • NSOperation可以调用start方法来执行任务,但默认是同步执行的

  • 如果将NSOperation添加到NSOperationQueue(操作队列)中,系统会自动异步执行NSOperation中的操作

  • 添加操作到NSOperationQueue中

-(void)addOperation:(NSOperation*)op;
-(void)addOperationWithBlock:(void(^)(void))block;
  • 代码:
  • 方法一:
    // 创建队列
    NSOperationQueue *queue = [[NSOperationQueue alloc] init];
    
    // 创建操作
        NSBlockOperation *op1 = [NSBlockOperation blockOperationWithBlock:^{
            NSLog(@"download1 --- %@", [NSThread currentThread]);
        }];
    
    //添加操作到队列中
        [queue addOperation:op1];
  • 方法二:
 // 创建队列
    NSOperationQueue *queue = [[NSOperationQueue alloc] init];
//添加操作到队列中
 [queue addOperationWithBlock:^{
        NSLog(@"download1 --- %@", [NSThread currentThread]);
    }];

NSOperationQueue的队列类型与GCD的队列类型对比

  • GCD的队列
  • 并发队列
    • 自己创建的
    • 全局
  • 串行队列
    • 主队列
    • 自己创建的
  • NSOperationQueue的队列
  • 主队列
    [NSOperationQueue mainQueue]
 - 凡是添加到主队列中的任务(NSOperation),都会放到主线程中执行
  • 非主队列(其他队列)
     [[NSOperationQueue alloc] init]
 - 同时包含了:串行、并发功能
 - 添加到这种队列中的任务(NSOperation),就会自动放到子线程中执行

最大并发数

  • 什么是并发数

  • 同时执行的任务数

  • 比如,同时开3个线程执行3个任务,并发数就是3

  • 最大并发数的相关方法

-(NSInteger)maxConcurrentOperationCount;
-(void)setMaxConcurrentOperationCount:(NSInteger)cnt;
  • 代码
     // 创建队列
     NSOperationQueue *queue = [[NSOperationQueue alloc] init]; 
     // 设置最大并发操作数
     // queue.maxConcurrentOperationCount = 2;// 并发队列
     queue.maxConcurrentOperationCount = 1; // 就变成了串行队列
     // 添加操作
     [queue addOperationWithBlock:^{
          NSLog(@"download1 --- %@", [NSThread currentThread])
      }];
     [queue addOperationWithBlock:^{
          NSLog(@"download2 --- %@", [NSThread currentThread])
      }];
     [queue addOperationWithBlock:^{
          NSLog(@"download3 --- %@", [NSThread currentThread])
      }];
    
    

队列的取消、暂停、恢复

  • 取消队列的所有操作
-(void)cancelAllOperations;

提示:也可以调用NSOperation的-(void)cancel方法取消单个操作

  • 暂停和恢复队列
  -(void)setSuspended:(BOOL)b;// YES代表暂停队列,NO代表恢复队列
  - (BOOL)isSuspended
  • 代码
 // 恢复队列,继续执行
 // self.queue.suspended = NO;

 // 暂停(挂起)队列,暂停执行
 // self.queue.suspended = YES;

 // 取消队列的所有操作
 [self.queue cancelAllOperations];

操作依赖

  • NSOperation之间可以设置依赖来保证执行顺序
  • 比如一定要让操作A执行完后,才能执行操作B,可以这么写
[operationB addDependency:operationA];// 操作B依赖于操作A
  • 也可以在不同queue的NSOperation之间创建依赖关系
  • 代码
    // 创建队列
    NSOperationQueue *queue = [[NSOperationQueue alloc] init];
    // 添加操作
    NSBlockOperation *op1 = [NSBlockOperation blockOperationWithBlock:^{
        NSLog(@"download1----%@", [NSThread  currentThread]);
    }];
    NSBlockOperation *op2 = [NSBlockOperation blockOperationWithBlock:^{
        NSLog(@"download2----%@", [NSThread  currentThread]);
    }];
    NSBlockOperation *op3 = [NSBlockOperation blockOperationWithBlock:^{
        NSLog(@"download3----%@", [NSThread  currentThread]);
    }];
   // 设置依赖(保证op3在op1和op2都执行完之后再执行)
    [op3 addDependency:op1];
    [op3 addDependency:op2];
  • 注意:不能相互依赖
  • 比如A依赖B,B依赖A

线程之间的通信

  • 举例:在子线程下载图片,再回答到主线程在imageView添加图片
  • 看代码:
   [[[NSOperationQueue alloc] init] addOperationWithBlock:^{
        // 图片的网络路径
        NSURL *url = [NSURL URLWithString:@"http://img.pconline.com.cn/images/photoblog/9/9/8/1/9981681/200910/11/1255259355826.jpg"];
        
        // 加载图片
        NSData *data = [NSData dataWithContentsOfURL:url];
        
        // 生成图片
        UIImage *image = [UIImage imageWithData:data];
        
        // 回到主线程
        [[NSOperationQueue mainQueue] addOperationWithBlock:^{
            self.imageView.image = image;
        }];
    }];

操作的监听

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

自定义NSOperation

  • 自定义NSOperation的步骤很简单
  • 重写-(void)main方法,在里面实现想执行的任务
  • 重写-(void)main方法的注意点
  • 自己创建自动释放池(因为如果是异步操作,无法访问主线程的自动释放池)
  • 经常通过-(BOOL)isCancelled方法检测操作是否被取消,对取消做出响应

你可能感兴趣的:(iOS多线程开发之NSOperation用法总结)