NSOperationQueue、NSOperation理解

1.NSThread与NSRunLoop:
    一个单独的app应用就是一个进程,里面起码包含有一个主线程(UI线程),并可以创建多个子线程
    每个thread中默认会有一个runLoop对象,主线程的runLoop对象是运行着的,用户自己创建的子线程runLoop对象默认是没有启动的。
    如果新开的子线程只是为了完成一个耗时的任务,那么不用启动runLoop循环,如果需要子线程监控一些事情,则需开启;只有给runLoop添加了源才能保证线程一直运行存在,这些源包括timer源、输入事件等,参考:http://www.hrchen.com/2013/06/multi-threading-programming-of-ios-part-1/ 和 http://blog.sina.com.cn/s/blog_74e9d98d0101gzxq.html

2.NSOperationQueue与NSOperation:
    一个operation就相当于一个函数块、block块、代码块,这里只是把它提高到一种任务的角度来看待,然后,任务便会有开始执行(start)、取消(cancel)、是否取消(isCancel)、是否完成(isFinishing)、暂停(pause)等状态函数,其本身是不会创建新的线程来执行它的,NSOperation本身是一个基类,不能直接使用,必须继承它。
    NSOperation中比较重要的是start和main函数,一般结合NSOperationQueue来使用,OperationQueue实质上也就是数组管理,对添加进去的operation进行管理、创建线程等;添加到queue里的operation,queue默认会调用operation的start函数来执行任务,而start函数默认又是调用main函数的。
    如果不想自定义operation,则可以简单使用系统预定义的两个operation子类:NSBlockOperation(任务块是一个block块)、NSInvocationOperation(任务块是一个函数块)
     另外,OperationQueue在使用上也有很大的缺陷,比如一旦operation添加到队列中后,就不能再对operation进行一些修改操作,否则将会出现不可预知的错误,因为operation在队列中有可能随时都会被执行;只能进行cancel或整个队列的cancelAllOperations操作,对单个operation进行cancel后,这个operation还会短暂停留在队列的operations数组里,要稍等片刻才会被移除,这样对于外界使用队列这个operation数组的情况将会出现bug,cancelAllOperations也是要等会才会全部被移除;值得一提的是,cancel和cancelAllOperations默认情况下,都不能取消已经正在执行的operation,只能取消在队列中等待的operation,如果硬要取消正在执行的,只能重写operation的main方法,并在main方法中添加对isCancel属性的监控;而且对于队列的setSuspended方法也只是暂停等待的operation,还有operation添加依赖、队列最大线程数等,功能都比较鸡肋,一切队列的方法都只能对等待的operation才能起作用。
    
    如此一来,特别对于像下载这样的,只能自定义operation对象,并重写start方法(start方法中,不一定仍然调用自身的main方法,可以自己指定执行的方法,像AFURLConnectionOperation一样)、自己创建线程;之前的operation本身是不会创建线程的,线程是在queue中创建,所以才产生并发的效果。
   
   自定义operation需要重写并注意的函数:网上说的那么多重写start main isExcuding实现KVO等  其实都是为了后面结合OperationQueue使用,如果你完全不想使用queue,则operation貌似也没有使用的必要,它本身就是一个代码块,不创建线程的,其start main函数都是为了提供给queue管理调用的,可参考AFURLConnectionOperation的实现
   NSOperation.h头文件:
@interface NSOperation : NSObject {
@private
    id _private;
    int32_t _private1;
#if __LP64__
    int32_t _private1b;
#endif
}

- (id)init; // designated initializer

- (void)start; //
- (void)main;

- (BOOL)isCancelled;
- (void)cancel;

- (BOOL)isExecuting;
- (BOOL)isFinished;

- (BOOL)isConcurrent;

- (BOOL)isReady;

- (void)addDependency:(NSOperation *)op;
- (void)removeDependency:(NSOperation *)op;

- (NSArray *)dependencies;

typedef NS_ENUM(NSInteger, NSOperationQueuePriority) {
    NSOperationQueuePriorityVeryLow = -8L,
    NSOperationQueuePriorityLow = -4L,
    NSOperationQueuePriorityNormal = 0,
    NSOperationQueuePriorityHigh = 4,
    NSOperationQueuePriorityVeryHigh = 8
};

- (NSOperationQueuePriority)queuePriority;
- (void)setQueuePriority:(NSOperationQueuePriority)p;

#if NS_BLOCKS_AVAILABLE
- (void (^)(void))completionBlock NS_AVAILABLE(10_6, 4_0);
- (void)setCompletionBlock:(void (^)(void))block NS_AVAILABLE(10_6, 4_0);
#endif

- (void)waitUntilFinished NS_AVAILABLE(10_6, 4_0);

- (double)threadPriority NS_AVAILABLE(10_6, 4_0);
- (void)setThreadPriority:(double)p NS_AVAILABLE(10_6, 4_0);

@end


posted @ 2013-12-27 11:49  SEC.VIP_网络安全服务  阅读(108)  评论(0编辑  收藏  举报