[Note] Effective OC - Item 43~45

Chapter 6. Blocks and Grand Central Dispatch



(昨天问大神死锁的问题的时候,大神推荐去看《Objective-C高级编程》这本,看了一下GCD部分觉得写得比这本清楚诶,后悔没早看!如果先看那本再看这本的第六章会比较好一点。)


Item 43: Know When to Use GCD and When to Use Operation



这一节讲GCD和NSOperationQueue用法上的区别,主要是NSOperationQueue比GCD更方便的地方。不过没有给代码例子。等我过几天补上。
GCD是调用c语言函数,NSOperation是OC对象,底层是用GCD来实现的,所以更重量级一些。虽然一般认为调用越底层的东西效率越高,作者的意见是也不能盲目迷信底层,还是要具体问题具体分析。
把文中内容和Reference合一下:

  • Operation Dependency: 涉及两个方法addDependency:removeDependency:,当一个operation依赖的最后一个operation完成的时候,这个operation才会被执行。这里的“完成”包括成功和不成功的情况。
  • KVO-Compliant Properties: NSOperation特别适合KVO监听,提供的监听属性很多,比如isCancelled, isAsynchronous, isExecuting, isFinished等等,这使得可以针对不同状态写不同的代码,处理就比GCD细致很多。
  • Multicore Considerations: NSOperation支持多核。在多个线程调用它的方法可以保证thread-safe。所以自定义NSOperation子类的时候,也要保证重写或者添加的方法要thread-safe。
  • Cancelling: 直接调用cancel方法就可以,但必须在任务运行之前,如果已经启动就不能取消了。
    Priority: NSOperation有单独针对某个操作的优先级,也有线程优先级,而GCD只有队列优先级,所以NSOperation的划分更加细致了。
  • Reuse of Operations:NSOperation本身是一个抽象类,有两个系统给的子类,NSInvocationOperation和NSBlockOperation,当然也可以自定义。

最后文中举了一个使用NSOperation做API的例子,NSNotificationCenter:

- (id)addObserverForName:(NSString *)name object:(id)object queue:(NEOperationQueue *)queue usingBlock:(void(^)(NSNotification *))block



Item 44: Use Dispatch Groups to Take Advantage of Platform Scaling



这一节讲dispatch group的用法。
Dispatch group主要用在需要等多个任务完成后再进行一个结束处理的情况(其实我有点纠结它和dispatch barrier的区别…如果把多个任务放到concurrent dispatch queue,然后把结束处理放到barrier里,感觉效果是差不多的?但是《Objective-C高级编程》这本里面说“虽然用dispatch group和dispatch_set_target_queue也可实现,但是源代码会很复杂”,我不是很懂),多个任务一般是采用并发队列,如果是串行队列好像就没什么意义。基本的用法不赘述了,这里我笔记两个细节:
多个任务不是一定要放到一个并发队列的,可以放到多个,而且可以分不同优先级。
dispatch_apply,按指定次数将相应的block追加到指定的dispatch queue中,并等待全部处理执行结束。用法:

void dispatch_apply(size_t iterations, dispatch_queue_t queue, void(^block)(size_t));

第一个参数重复次数,第二个是追加到的queue,第三个就是追加的block。需要注意的是dispatch_apply和dispatch_sync一样会等待处理执行结束,容易引起死锁,所以推荐在dispatch_async函数中非同步地执行dispatch_apply函数。像这样:

dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);
dispatch_async(queue, ^{
   dispatch_apply(…);
});



Item 45: Use dispatch_once for Thread-Safe Single-Time Code Execution



这一节讲dispatch_once。这一节好短啊!不过我喜欢,2333333

void dispatch_once(dispatch_once_t *token, dispatch_block_t block);

这是一个保证在应用程序执行中block只执行一次的函数,完全的thread-safe,而且还很高效。第一个参数token一般声明为静态,来保证每次调用函数时看到的token是一致的。以实现单例的sharedInstance方法为例:

+ (id)sharedInstance {
    static EOCClass *sharedInstance = nil;
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        sharedInstance = [[self alloc] init];
    });
    return sharedInstance;
}

你可能感兴趣的:([Note] Effective OC - Item 43~45)