NSOperation和GCD一样,不用我们管理线程的生命周期,加锁等问题,只要把操作封装进NSOperation中,系统会自动帮我们创建线程,执行操作。而且他是面向对象的,我们看起来更容易理解,使用起来也更灵活。GCD提供的API都是C语言的,看起来确实有点头痛。
NSOperation是一个抽象类,我们得使用他的两个子类NSInvocationOperation和NSBlockOperatio才能实现多线程,当然我们也可以自定义。那下面就先介绍一下该怎么使用。
NSInvocationOperation
代码
- (void)viewDidLoad {
[super viewDidLoad];
// 初始化一个对象,并把操作(相当于GCD里的任务)封装进去
NSInvocationOperation *operation = [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(run) object:nil];
// 开始执行 一开始就执行run方法
[operation start];
NSLog(@"怎么才执行我");
}
- (void)run {
[NSThread sleepForTimeInterval:1.0];
NSLog(@"执行操作%@",[NSThread currentThread]);
}
日志
2016-11-06 08:23:38.999 TTTTTTTTTT[2872:38936] 执行操作<NSThread: 0x7a710ca0>{number = 1, name = main}
2016-11-06 08:23:38.999 TTTTTTTTTT[2872:38936] 怎么才执行我
分析:
(1)操作默认在主线程中执行,看打印的第一条日志。
(2)是同步执行的,会堵塞当前线程,看第二条日志,run方法执行完,“怎么才执行我”才打印。
NSBlockOperation
代码
- (void)viewDidLoad {
[super viewDidLoad];
// 初始化,并把要执行的操作封装进block中
NSBlockOperation *operation = [NSBlockOperation blockOperationWithBlock:^{
[NSThread sleepForTimeInterval:1.0];
NSLog(@"执行操作%@",[NSThread currentThread]);
}];
[operation start];
NSLog(@"我知道我想早点执行,那是不可能的");
}
日志
2016-11-06 08:39:16.815 TTTTTTTTTT[3528:48184] 执行操作<NSThread: 0x60800006e4c0>{number = 1, name = main}
2016-11-06 08:39:16.816 TTTTTTTTTT[3528:48184] 我知道我想早点执行,那是不可能的
分析:是不是看起来和NSInvocationOperation一样,都是同步在主线程中执行,其实NSBlockOperation是可以并发执行的;
NSBlockOperation的并发执行
代码
- (void)viewDidLoad {
[super viewDidLoad];
// 初始化,并把要执行的操作封装进block中
NSBlockOperation *operation = [NSBlockOperation blockOperationWithBlock:^{
// 添加任务A
NSLog(@"执行操作A%@",[NSThread currentThread]);
}];
[operation addExecutionBlock:^{
// 添加任务B
NSLog(@"执行操作B%@",[NSThread currentThread]);
}];
[operation addExecutionBlock:^{
// 添加任务C
NSLog(@"执行操作C%@",[NSThread currentThread]);
}];
[operation start];
}
日志
2016-11-06 08:43:48.102 TTTTTTTTTT[3707:50733] 执行操作B<NSThread: 0x60000007f200>{number = 3, name = (null)}
2016-11-06 08:43:48.102 TTTTTTTTTT[3707:50702] 执行操作A<NSThread: 0x60000007cac0>{number = 1, name = main}
2016-11-06 08:43:48.102 TTTTTTTTTT[3707:50734] 执行操作C<NSThread: 0x608000268840>{number = 4, name = (null)}
分析:
(1)看到没有,操作B和C都是在子线程执行的,实现了异步并发。
(2)当NSBlockOperation里的要执行的操作的数量 >1的时,就会异步并发执行;否则(当NSBlockOperation里的要执行的操作只有一个),就会默认在主线程中同步执行。
NSOperationQueue
NSInvocationOperation里面的操作也想异步并发执行可以不可以呢?当然没问题了,用NSOperationQueue 就行。
代码
- (void)viewDidLoad {
[super viewDidLoad];
// 初始化一个NSOperationQueue对象
NSOperationQueue *queue= [NSOperationQueue new];
NSInvocationOperation *operation1 = [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(run1) object:nil];
// 添加操作
[queue addOperation:operation1];
NSInvocationOperation *operation2 = [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(run2) object:nil];
// 添加操作
[queue addOperation:operation2];
// 还可以这样添加操作
[queue addOperationWithBlock:^{
NSLog(@"执行第3个操作%@",[NSThread currentThread]);
}];
}
- (void)run1 {
NSLog(@"执行第1个操作%@",[NSThread currentThread]);
}
- (void)run2 {
NSLog(@"执行第2个操作%@",[NSThread currentThread]);
}
日志
2016-11-06 08:57:03.498 TTTTTTTTTT[4195:57383] 执行第2个操作<NSThread: 0x60800006f8c0>{number = 3, name = (null)}
2016-11-06 08:57:03.498 TTTTTTTTTT[4195:57384] 执行第1个操作<NSThread: 0x608000079480>{number = 4, name = (null)}
2016-11-06 08:57:03.498 TTTTTTTTTT[4195:57386] 执行第3个操作<NSThread: 0x600000079e40>{number = 5, name = (null)}
分析:
(1)三个操作分别在不同的线程中执行,实现了并发。
(2)添加进队列的任务会自动执行,不要我们开启了。 [operation start];
不用写了。
(3)添加进队列的任务也是遵循先进先出的FIFO准则。那有人就要问了,为什么第一个操作不是第一个先执行完?这其实和100米赛跑一个起跑的不是第一个到终点一个道理,不矛盾。
总结
NSOperation的使用步骤和GCD没啥区别,都是确定要执行的操作,把操作放进NSOperation中,开始执行。如果想异步并发操作,在加一步,把NSOperation放进队列中。记住一点不能直接使用NSOperation,要使用它的两个子类NSInvocationOperation和NSBlockOperatio。
其实关于NSOperation还有很多基本的属性和方法,下一篇文章再讲。看到NSOperation的使用是不是舒服多了……