014-NSOperation多线程技术

NSOperation 是用 OC 语言对 GCD 的封装,面向对象,使用简单,并且提供了一些额外的操作,比如取消任务、添加任务间依赖关系等,不需要关心线程和线程生命周期,自动多线程执行。

NSOperation 是一个抽象类,使用 NSOperation 有三种方法

  • NSInvocationOperation
  • NSBlockOperation
  • 自定义子类继承 NSOperation

NSInvocationOperation

使用 NSInvocationOperation 需要自定义执行函数

- (void)operationTestLearn
{
    NSInvocationOperation *operation = [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(task:) object:@"task"];
    [operation start];
}

- (void)task:(NSString *)value
{
    NSLog(@"%@", value);
}

默认情况下执行 start 方法是在当前线程同步执行操作,如果要异步多线程执行,则要用 NSOperationQueue。

    NSInvocationOperation *operation = [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(task:) object:@"task"];
    NSOperationQueue *queue = [NSOperationQueue new];
    [queue addOperation:operation];

NSBlockOperation

NSBlockOperation 使用 block 作为需要执行的操作。

    NSBlockOperation *operation = [NSBlockOperation blockOperationWithBlock:^{
        NSLog(@"block task");
    }];
    [operation start];

单独执行一个 block 是同步操作,但 NSBlockOperation 支持加入多个 block,这样就会进行异步多线程执行操作。

    NSBlockOperation *operation = [NSBlockOperation blockOperationWithBlock:^{
        NSLog(@"block task1");
    }];
    
    [operation addExecutionBlock:^{
        NSLog(@"block task2");
    }];
    
    [operation addExecutionBlock:^{
        NSLog(@"block task3");
    }];
    
    [operation addExecutionBlock:^{
        NSLog(@"block task4");
    }];
    
    [operation start];

这里每一个 block 都会在单独一个线程内执行。

当然也可以将 operation 放入队列中执行。

    NSBlockOperation *operation = [NSBlockOperation blockOperationWithBlock:^{
        NSLog(@"block task1");
    }];
    
    [operation addExecutionBlock:^{
        NSLog(@"block task2");
    }];
    
    [operation addExecutionBlock:^{
        NSLog(@"block task3");
    }];
    
    [operation addExecutionBlock:^{
        NSLog(@"block task4");
    }];
    
    NSOperationQueue *queue = [NSOperationQueue new];
    [queue addOperation:operation];

NSOperationQueue 也有执行在主线程的 main queue,是一个串行队列

NSOperationQueue *queue = [NSOperationQueue mainQueue];

自定义 NSOperation

继承 NSOperation,实现其中的 main 方法可以简单实现一个 operation,当然也可以复写 start 方法,但是要手动管理状态

需要手动管理的状态有:

  • isExecuting 代表任务正在执行中
  • isFinished 代表任务已经执行完成
  • isCancelled 代表任务已经取消执行

必须通过 KVO 机制通知状态转移,由于 operation 支持 cancel,所以 isCancelled 用于检查是否需要取消当前操作。当实现了start方法时,默认会执行start方法,而不执行main方法。

线程相关

队列可以设置最大并发数

    NSLog(@"queue maxConcurrentOperationCount %ld", [queue maxConcurrentOperationCount]);
    [queue setMaxConcurrentOperationCount:4];

可以取消、暂停或恢复操作。

    [queue cancelAllOperations];
    [queue setSuspended:YES];

当然 NSOperation 也可以单独取消

    [operation cancel];

还可以对 operation 添加依赖关系,这种关系可以跨线层依赖

    NSBlockOperation *operation1 = [NSBlockOperation blockOperationWithBlock:^{
        NSLog(@"block task1");
    }];
    
    NSBlockOperation *operation2 = [NSBlockOperation blockOperationWithBlock:^{
        NSLog(@"block task2");
    }];
    
    [operation2 addDependency:operation1];
    
    NSBlockOperation *operation3 = [NSBlockOperation blockOperationWithBlock:^{
        NSLog(@"block task3");
    }];
    
    [operation3 addDependency:operation2];
    
    NSBlockOperation *operation4 = [NSBlockOperation blockOperationWithBlock:^{
        NSLog(@"block task4");
    }];
    
    [operation4 addDependency:operation3];
    
    NSOperationQueue *queue = [NSOperationQueue mainQueue];
    [queue addOperation:operation1];
    [queue addOperation:operation2];
    [queue addOperation:operation3];
    [queue addOperation:operation4];

但是依赖关系不能形成环,否则都无法执行。

如果想监听 operation 是否执行完,也可以加上监听

    NSBlockOperation *operation = [NSBlockOperation blockOperationWithBlock:^{
        NSLog(@"running");
    }];
    
    [operation setCompletionBlock:^{
        NSLog(@"finished");
    }];
    
    NSOperationQueue *queue = [NSOperationQueue new];
    [queue addOperation:operation];

你可能感兴趣的:(014-NSOperation多线程技术)