使用NSOperation以及NSOperationQueue
NSOperation vs. Grand Central Dispatch (GCD)
在Mac OS X v10.6和iOS4之前,NSOperation 与 NSOperationQueue 不同于GCD,他们使用了完全不同的机制。
从Mac OS X v10.6和iOS4开始,NSOperation 和 NSOperationQueue是建立在GCD上的。作为一种通例,苹果推荐使用最高级别的抽象,然而当评估显示有需要时,会突然降到更低级别。
以下是对两者的快速比较,它会帮助你决定何时何地去使用GCD或者NSOperation和NSOperationQueue;
- GCD是一种轻量级的方法来代表将要被并发执行的任务单位。你并不需要去计划这些任务单位;系统会为你做计划。在块(block)中添加依赖会是一件令人头疼的事情。取消或者暂停一个块会给一个开发者产生额外的工作!:]
- NSOperation和NSOperationQueue 对比GCD会带来一点额外的系统开销,但是你可以在多个操作(operation)中添加附属。你可以重用操作,取消或者暂停他们。NSOperation和 Key-Value Observation (KVO)是兼容的;例如,你可以通过监听NSNotificationCenter去让一个操作开始执行。
NSOperation API
NSOperation 类有一个相当简短的声明。要定制一个操作,可以遵循以下步骤:
- 继承NSOperation类
- 重写“main”方法
- 在“main”方法中创建一个“autoreleasepool”
- 将你的代码放在“autoreleasepool”中(ARC中也需要放在autoreleasepool中,这是为了兼容MRC代码)
为了你能够测试代码,请新建一个类,继承自NSOperation,写成如下的形式:
之后再进行操作:
注意,只有将一个任务添加进了队列当中,这个任务才会被执行.
以下代码是并发执行2个任务,队列的最大并发数是4:
任务具有的优先级:
当你添加了操作到一个队列时,在对操作调用“start”方法之前,NSOperationQueue会浏览所有的操作。那些有较高优先级的操作会被先执行。有同等优先级的操作会按照添加到队列中的顺序去执行(先进先出)。
(历史注释:在1997年,火星车中的嵌入式系统遭遇过优先级反转问题,也许这是说明正确处理优先级和互斥锁的最昂贵示例了。想对这一事件的背景知识有更多的了解,可以看这个网址: http://research.microsoft.com/en-us/um/people/mbj/Mars_Pathfinder/Mars_Pathfinder.html )
请添加如下的延迟方法:
先开始任务,1s钟之后cancel掉一个任务:
每个任务结束后都会有一个回调可以表示任务结束:
任务之间的从属关系:
以下来点总结:
- 要经常检查isCancelled属性。如果操作不需要被执行了,你就不想在后台去运行它了!(你从外面设置了cancell后,你是需要在方法里面来实现cancell的哦,注意,亲)
- 你并不需要重写“start”方法。然而,如果你决定去重写“start”方法,就必须处理好像isExecuting, isFinished, isConcurrent 和 isReady这些属性。否则你的操作类不会正确的运作。
- 你一旦添加了一个操作到一个队列(NSOperationQueue的一个实例)中,就要负责释放它(如果你不使用ARC的话)。NSOperationQueue 获得操作对象的所有权,调用“start”方法,然后结束时负责释放它。
- 你不能重用一个操作对象。一旦它被添加到一个队列中,你就丧失了对它的所有权。如果你想再使用同一个操作类,就必须创建一个新的实例变量。
- 一个结束的操作不能被重启。
- 如果你取消了一个操作,它不会马上就发生。它会在未来的某个时候某人在“main”函数中明确地检查isCancelled == YES 时被取消掉;否则,操作会一直执行到完成为止。(见上图)
- 一个操作是否成功地完成,失败了,或者是被取消了,isFinished的值总会被设置为YES。所以千万不要觉得isFinished == YES就表示所有的事情都顺利完成了 — 特别的,如果你在代码里面有从属性(dependencies),就要更加注意!
NSOperationQueue API
查看当前并发操作的所有任务:
挂起与恢复操作:
This method suspends or resumes the execution of operations. Suspending a queue prevents that queue from starting additional operations. In other words, operations that are in the queue (or added to the queue later) and are not yet executing are prevented from starting until the queue is resumed. Suspending a queue does not stop operations that are already running.
这个方法会挂起或者恢复一个执行的任务.挂起一个队列可以阻止该队列中没有开始的任务.换句话说,在任务队列中还没有开始执行的任务是会被挂起的,直到这个挂起操作被恢复.挂起一个队列不会让一个已经执行的任务停止哦,亲.
取消所有操作:
要取消一个队列中的所有操作,你只要简单的调用“cancelAllOperations”方法即可。还记得之前提醒过经常检查NSOperation中的isCancelled属性吗?
原因是“cancelAllOperations”并没有做太多的工作,他只是对队列中的每一个操作调用“cancel”方法 — 这并没有起很大作用!:] 如果一个操作并没有开始,然后你对它调用“cancel”方法,操作会被取消,并从队列中移除。然而,如果一个操作已经在执行了,这就要由单独的操作去识 别撤销(通过检查isCancelled属性)然后停止它所做的工作。
简单的block形式的队列任务:
以上就基本上讲完了怎么使用NSOperation以及NSOperationQueue了,亲.