多线程之NSOperationQueue,与GCD的对比

2017-06-30  本文已影响146人  大猿媛

NSOperationQueue与GCD是伯仲之间吗

事实:GCD是面向底层的C语言的API, NSOperationQueue是基于GCD面向OC的封装
结果:
1)GCD执行效率更高
2)GCD只支持FIFO的队列,而NSOperationQueue可以通过设置最大并发数,设置优先级,添加依赖关系等调整执行顺序
3)NSOperationQueue甚至可以跨队列设置依赖关系,但是GCD只能通过设置串行队列,或者在队列内添加barrier(dispatch_barrier_async)任务,才能控制执行顺序,较为复杂

  1. NSOperationQueue因为面向对象,所以支持KVO,可以监测operation是否正在执行(isExecuted)、是否结束(isFinished)、是否取消(isCanceld)
NSOperationQueue的API
    //start 开启任务,默认主队列
    NSBlockOperation *operationOne = [NSBlockOperation blockOperationWithBlock:^{\
        for (int i=0; i<3; i++) {
            NSLog(@"=%@=====%d",[NSThread currentThread],i);
        }
    }];
    [operationOne start];
   //主队列
    NSOperationQueue *mainqueue = [NSOperationQueue mainQueue];

    //自己创建的队列
    NSOperationQueue *customQueue = [[NSOperationQueue alloc]init];
    customQueue.maxConcurrentOperationCount = 2; //最大并发数

    //只要是自己创建的队列,就会在子线程中执行,默认并发,设置maxConcurrentOperationCount时是串行执行
    [customQueue addOperationWithBlock:^{
       //向队列中添加block异步任务
        for (int i=3; i<6; i++) {
            NSLog(@"=%@=====%d",[NSThread currentThread],i);

        }
    }];
    
    [customQueue addOperationWithBlock:^{
        //向队列中添加block异步任务
        for (int i=6; i<9; i++) {
            NSLog(@"=%@=====%d",[NSThread currentThread],i);
        }
    }];
    
    //添加 NSInvocationOperation
    [customQueue addOperation:[[NSInvocationOperation alloc]initWithTarget:self selector:@selector(invocationOperationDeal) object:nil]];
    
    //将队列中任务暂停
    customQueue.suspended = YES;
    
    //将队列中任务恢复执行
    customQueue.suspended = NO;
    
    //取消队列中的所有任务
    [customQueue cancelAllOperations];
    //任务之间添加依赖,比如当前一个任务的返回结果  是 后一个任务的所需元素时
    NSBlockOperation *blockOp1 = [NSBlockOperation blockOperationWithBlock:^{
       //计算出小明的脸部数据
    }];
    
    NSBlockOperation *blockOp2 = [NSBlockOperation blockOperationWithBlock:^{
       //计算出小明的头围数据
    }];
    
    NSBlockOperation *blockOp3 = [NSBlockOperation blockOperationWithBlock:^{
       // 制作出小明的头部模型
    }];
    [blockOp3 addDependency:blockOp1];
    [blockOp3 addDependency:blockOp2];  //blockOp3的操作依赖于 1 和 2 的结果
    [customQueue addOperations:@[blockOp1,blockOp2,blockOp3] waitUntilFinished:NO];
[a setQueuePriority:NSOperationQueuePriorityVeryLow];
一共有四个优先级:
NSOperationQueuePriorityLow,
NSOperationQueuePriorityNormal,
NSOperationQueuePriorityHigh,
NSOperationQueuePriorityVeryHigh
 [blockOp2 setCompletionBlock:^{
        //blockOp2任务执行完成,此block在主线程执行
 }];
小结

1、实际项目开发中,很多时候只是会用到异步操作,不会有特别复杂的线程关系管理,所以苹果推崇的且优化完善、运行快速的GCD是首选

2、如果涉及到类似多线程并发下载,这种多任务还要控制任务进度的情况,可能就要考虑使用易于管理的NSOperationQueue了,关于自定义NSOperation的内容,后续我会另写一篇文章的,继续学习

3、另外不论是GCD还是NSOperationQueue,我们接触的都是任务和队列,都没有直接接触到线程这个东东,事实上线程管理也的确不需要我们操心,系统对于线程的创建,调度管理和释放都做得很好。但是有一个线程类NSThread可以满足我们对于线程的好奇心,但是NSThread需要我们自己去管理线程的生命周期,还要考虑线程同步、加锁问题,造成一些性能上的开销

附图,这张图如此之精辟,让我久久挪不开眼睛


03C5DAFC-D70B-4D8B-8805-41711EBDEC1C.png
上一篇 下一篇

猜你喜欢

热点阅读