NSOperation的作用
配合使用NSOperation和NSOperationQueue也能实现多线程编程
NSOperation和NSOperationQueue实现多线程的具体步骤:
先将需要执行的操作封装到一个NSOperation对象中
然后将NSOperation对象添加到NSOperationQueue中
系统会自动将NSOperationQueue中的NSOperation取出来
将取出的NSOperation封装的操作放到一条新线程中执行
NSOperation是个抽象类,并不具备封装操作的能力,必须使用它的子类
使用NSOperation子类的方式有3种:
NSInvocationOperation
NSBlockOperation
自定义子类继承NSOperation,实现内部相应的方法
NSInvocationOperation
创建NSInvocationOperation对象
- (id)initWithTarget:(id)target selector:(SEL)sel object:(id)arg;
调用start方法开始执行操作
- (void)start;
一旦执行操作,就会调用target的sel方法
注意:
默认情况下,调用了start方法后并不会开一条新线程去执行操作,而是在当前线程同步执行操作
只有将NSOperation放到一个NSOperationQueue中,才会异步执行
// 1.创建操作对象, 封装要执行的任务
NSInvocationOperation *operation = [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(download) object:nil];
// 2.执行操作(默认情况下, 如果操作没有放到队列queue中, 都是同步执行)
[operation start];
- (void)download
{
for (int i = 0; i<10; i++) {
NSLog(@"------op---%@", [NSThread currentThread]);
[NSThread sleepForTimeInterval:0.2];
}
}
NSBlockOperation
创建NSBlockOperation对象
+ (id)blockOperationWithBlock:(void (^)(void))block;
通过addExecutionBlock:方法添加更多的操作
- (void)addExecutionBlock:(void (^)(void))block;
- 注意:只要NSBlockOperation封装的操作数 > 1,就会异步执行操作
// 1.封装操作
NSBlockOperation *operation = [[NSBlockOperation alloc] init];
[operation addExecutionBlock:^{
NSLog(@"NSBlockOperation----op1---%@", [NSThread currentThread]);
}];
[operation addExecutionBlock:^{
NSLog(@"NSBlockOperation-----op2--%@", [NSThread currentThread]);
}];
[operation addExecutionBlock:^{
NSLog(@"NSBlockOperation-----op3--%@", [NSThread currentThread]);
}];
// 2.执行操作
[operation start];
自定义NSOperation
自定义NSOperation的步骤: (了解基本流程)
重写- (void)main方法,在里面实现想执行的任务
重写- (void)main方法的注意点:
自己创建自动释放池(因为如果是异步操作,无法访问主线程的自动释放池)
经常通过- (BOOL)isCancelled方法检测操作是否被取消,对取消做出响应
NSOperationQueue
NSOperationQueue的作用:
NSOperation可以调用start方法来执行任务,但默认是同步执行的 如果将NSOperation添加到NSOperationQueue(操作队列)中,系统会自动异步执行NSOperation中的操作
添加操作到NSOperationQueue中:
- (void)addOperation:(NSOperation *)op;
- (void)addOperationWithBlock:(void (^)(void))block;
- 最大并发数
最大并发数: 同时执行的任务数 比如,同时开3个线程执行3个任务,并发数就是3
最大并发数的相关方法:
- (NSInteger)maxConcurrentOperationCount;
- (void)setMaxConcurrentOperationCount:(NSInteger)cnt;
- 操作优先级
设置NSOperation在queue中的优先级,可以改变操作的执行优先级
- (NSOperationQueuePriority)queuePriority;
- (void)setQueuePriority:(NSOperationQueuePriority)p;
优先级的取值
NSOperationQueuePriorityVeryLow = -8L,
NSOperationQueuePriorityLow = -4L,
NSOperationQueuePriorityNormal = 0,
NSOperationQueuePriorityHigh = 4,
NSOperationQueuePriorityVeryHigh = 8
操作依赖
NSOperation之间可以设置依赖来保证执行顺序
[operationB addDependency:operationA]; // 操作B依赖于操作A
可以在不同queue的NSOperation之间创建依赖关系,即依赖关系是可以跨队列的
如图:注意:队列操作之间不能互相依赖,比如: A依赖于B, B依赖于A,会造成循环引用
队列的取消、暂停、恢复
取消队列的所有操作:
- (void)cancelAllOperations;
提示:也可以调用NSOperation的- (void)cancel方法取消单个操作
暂停和恢复队列
- (void)setSuspended:(BOOL)b; // YES代表暂停队列,NO代表恢复队列
- (BOOL)isSuspended;
下面是代码实现:
#import "ViewController.h"
@interface ViewController ()
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
// NSOperation
NSInvocationOperation *invocationOP1 = [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(download) object:nil];
NSInvocationOperation *invocationOP2 = [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(run) object:nil];
// NSBlockOperation
NSBlockOperation *opration = [NSBlockOperation blockOperationWithBlock:^{
for (int i = 0; i < 10; i++) {
NSLog(@"NSBlockOperation--------%@", [NSThread currentThread]);
// 线程休眠0.2
[NSThread sleepForTimeInterval:0.2];
}
}];
// 创建队列
NSOperationQueue *queue = [[NSOperationQueue alloc] init];
// 设置最大并发数 (通常最大并发数2~3)
queue.maxConcurrentOperationCount = 2;
// 设置操作依赖,为了操作有序进行 执行顺序: opration > invocationOP2 > invocationOP1
[invocationOP1 addDependency:invocationOP2];
[invocationOP2 addDependency:opration];
// 将操作添加到队列中(自动执行,自动开启线程)
[queue addOperation:invocationOP1];
[queue addOperation:invocationOP2];
[queue addOperation:opration];
}
#pragma mark - 监听方法
- (void)download
{
for (int i = 0; i < 10; i++) {
NSLog(@"invocationOP1--------%@", [NSThread currentThread]);
// 线程休眠0.2
[NSThread sleepForTimeInterval:0.2];
}
}
- (void)run
{
for (int i = 0; i < 10; i++) {
NSLog(@"invocationOP2--------%@", [NSThread currentThread]);
// 线程休眠0.2
[NSThread sleepForTimeInterval:0.2];
}
}
@end