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];