Operation
Operation 是一个抽象的线程类, 可以直接使用系统提供的子类(NSInvocation 或者 BlockOperation) 去执行多线程任务. Operation 是线程安全的, 我们在使用 Operation 的时候只需要把注意力放到线程任务的实现上, 无需考虑线程安全
operation 对象 是一次性对象, 每次执行完任务后 operation 都会被销毁.
operation 的执行
- 将 operation 加入到 operationQueue 中异步去执行任务
- 手动调用
start()
来直接同步立刻执行任务, 这里有风险,如果 operation 不是处于 ready 状态 手动启动会抛出异常.
operation 的依赖
依赖关系决定了线程之间的执行顺序, 线程A 依赖线程B 则 要等到线程B 执行完了之后 线程A 才会执行. addDependency(_:)
添加依赖, removeDependency(_:)
删除依赖, 只有所有的依赖线程执行完, 这个线程才处于 ready 状态, 可以开始执行
依赖关系并不关心线程任务是否成功执行(也就是说, cansel 和 finish 对依赖关系来说是一样的). 他取决于你 在线程任务失败或者线程取消后是否继续执行.
遵从 KVO KVC
的属性
isCancelled - read-only //是够取消
isAsynchronous - read-only //是否异步
isExecuting - read-only //是否执行
isFinished - read-only //是否结束
isReady - read-only //是否准备好
dependencies - read-only //依赖
queuePriority - readable and writable //优先级
completionBlock - readable and writable //事件回调
多线程注意事项
NSOperation 类 本身是线程安全的
但是, NSOperation 的子类(当我们创建时) , 重写父类的方法是线程安全的, 但是 自定义方法要确保线程安全 , 所有修改变量的自定义方法 都要同步进行,防止数据异常.
同步和异步线程
手动执行 Operation 而不是把 Operation 加入到 队列中, 你可以让Operation 同步或者异步运行, 默认是同步. 调用start()
方法后 开始执行任务, 同步操作进行后, 对象的控制器返回给调用者
当你采用异步方式去调用 start()
, start()
方法可能在 线程任务还没有执行完就已经返回, 即使 start()
方法已经结束, 异步线程仍可以继续执行
将 operation 加入队列 和 手动异步执行 operation是一样的, 手动异步执行需要做更多的工作,你需要使用 kvo 监控线程状态, 当你不想使用 block 执行异步任务时, 手动异步执行 operation 是很好的代替者
将 operation 加入队列会自动异步调用 start()
方法, 忽略掉 operation 属性 isAsynchronous
, 很明显这时候是一个异步任务
operation 的子类
非并发子类
只需要重写一个方法 main()
你需要在 main()
方法中执行任务
并发子类
需要重写下面的方法
start()
isAsynchronous
isExecuting
isFinished
start()
负责启动异步操作, 在 start()
方法中要跟新 isExecuting
的状态, 可以使用 KVO 来监控状态. 需要在线程安全的条件下提供 isExecuting
状态
当异步任务完成或者取消之后, 你的并发炒作对象 operation 必须更新isExecuting
和 isFinished
状态.
(在取消的情况下,更新isFinished关键路径仍然很重要,即使操作未完成任务,排队的操作必须报告完成,然后才能从队列中删除。)除了 生成KVO通知,您的isExecuting和isFinished属性的覆盖也应该继续根据您的操作状态报告准确的值。
注意:
不应该调用父类的 start()
方法, 在重写 start()
方法时先判断该任务是否已经被取消.