NSOperation 和 NSOperationQueue 是苹果提供的一套多线程解决方案,是基于 GCD 的封装,完全面向对象,代码可读性较高。
NSOperation
NSOperation 是一个抽象类,并不具备封装操作的能力,用作父类用来约束子类,要想封装操作,必须使用它的子类。
我们可以使用系统提供的两个子类 NSBlockOperation 和 NSInvocationOperation,或者我们也可以自定义一个类,继承自 NSOperation,然后在自定义的类的 main 函数中实现具体操作。
默认情况下,单独使用 NSOperation 会在当前线程中同步执行操作,只有配合使用 NSOperationQueue 才能能更好的实现操作的异步执行。
NSInvocationOperation
NSInvocationOperation 初始化方法有两个,分别是:
// 该方法中的 id 类型参数 arg 可以传递到 sel 方法中。
- (nullable instancetype)initWithTarget:(id)target selector:(SEL)sel object:(nullable id)arg;
- (instancetype)initWithInvocation:(NSInvocation *)inv;
1、- (nullable instancetype)initWithTarget:(id)target selector:(SEL)sel object:(nullable id)arg;
NSLog(@"NSInvocationOperation begin");
NSInvocationOperation *operation = [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(operationTest:) object:@"test"];
[operation start];
NSLog(@"NSInvocationOperation end");
- (void)operationTest:(id)obj {
NSLog(@"task -- obj: %@, current queue: %@", obj, [NSThread currentThread]);
}
运行后查看打印信息:
2019-09-16 13:58:59.614875+0800 NSOperationSummary[66718:2581783] NSInvocationOperation begin
2019-09-16 13:58:59.616275+0800 NSOperationSummary[66718:2581783] task -- obj: test, current queue: <NSThread: 0x6000025ee900>{
number = 1, name = main}
2019-09-16 13:58:59.616440+0800 NSOperationSummary[66718:2581783] NSInvocationOperation end
说明单独使用 NSInvocationOperation 会在 当前线程中 同步 执行操作。
2、- (instancetype)initWithInvocation:(NSInvocation )inv;
NSLog(@"NSInvocationOperation begin");
NSMethodSignature *signature = [[self class] instanceMethodSignatureForSelector:@selector(operationTest:)];
NSInvocation *invocation = [NSInvocation invocationWithMethodSignature:signature];
invocation.target = self;
invocation.selector = @selector(operationTest:);
NSInvocationOperation *operation = [[NSInvocationOperation alloc] initWithInvocation:invocation];
operation start];
NSLog(@"NSInvocationOperation end");
运行后查看打印信息:
2019-09-16 14:04:33.100057+0800 NSOperationSummary[66809:2584211] NSInvocationOperation begin
2019-09-16 14:04:33.100746+0800 NSOperationSummary[66809:2584211] task -- obj: (null), current queue: <NSThread: 0x600002b19480>{
number = 1, name = main}
2019-09-16 14:04:33.100934+0800 NSOperationSummary[66809:2584211] NSInvocationOperation end
可以看到运行结果和第一种方法几乎没有区别,两种方法的唯一区别就是第一种方法可以传递数据到操作的方法中
我们看到:NSInvocationOperation 实例对象直接调用 start 方法是在当前线程执行封装的操作,而不是在子线程中执行。也就是说,NSInvocationOperation 实例对象直接调用 start 方法不会开启新线程异步执行,而是同步执行。只有将 NSInvocationOperation 实例对象添加到一个 NSOperationQueue 队列中,才会异步执行操作。
NSBlockOperation
NSBlockOperation 是 NSOperation 的子类。NSBlockOperation 中给我们提供了两个方法:
+ (instancetype)blockOperationWithBlock:(void (^)(void))block;
- (void)addExecutionBlock:(void (^)(void))block;
第一个是类方法,可以通过类方法直接初始化一个 NSBlockOperation 对象。第二个是实例方法,可以给一个已经存在的 NSBlockOperation 对象添加额外的操作。
和 NSInvocationOperation 相比,NSBlockOperation 对象不用添加到操作队列也能开启新线程,但是开启新线程是有条件的。前提是一个 NSBlockOperation 对象需要封装多个操作。
下面例子中,NSBlockOperation 对象只有一个操作,默认会在当前线程执行:
NSLog(@"NSBlockOperation begin");
NSBlockOperation *operation = [NSBlockOperation blockOperationWithBlock:^{
NSLog(@"task -- %@", [NSThread currentThread]);
}];
[operation start];
NSLog(@"NSBlockOperation end");
运行后查看打印信息:
2019-09-16 14:10:08.479299+0800 NSOperationSummary[66894:2586699] NSBlockOperation begin
2019-09-16 14:10:08.479726+0800 NSOperationSummary[66894:2586699] task -- <NSThread: 0x600003db96c0>{
number = 1, name = main}
2019-09-16 14:10:08.479863+0800 NSOperationSummary[66894:2586699] NSBlockOperation end
可以看到,操作是在主线程中同步执行的。
下面我们测试一下封装多个操作后的执行情况:初始化一个 NSBlockOperation 对象,然后调用 addExecutionBlock: 方法给这个 NSBlockOperation 对象添加多个操作。
NSLog(@"NSBlockOperation begin");
NSBlockOperation *operation = [NSBlockOperation blockOperationWithBlock:^{
NSLog(@"task 1 -- %@", [NSThread currentThread]);
}];
[operation addExecutionBlock:^{
NSLog(@"task 2 -- %@", [NSThread currentThread]);
}];
[operation addExecutionBlock:^{
NSLog(@"task 3 -- %@", [NSThread currentThread]);
}];
[operation addExecutionBlock:^{
NSLog(@"task 4 -- %@", [NSThread currentThread]);
}];
operation start];
NSLog(@"NSBlockOperation end");
运行后查看打印信息:
2019-09-16 14:12:19.959954+<