typealias KVONotificationBlock = (Any?, _ oldValue: Any?, _ value: Any?) -> Void
extension NSObject {
//默认的函数,option的初始值是Initial|New, 监测打变化的值默认转到主线程
func observe(_ object: Any?, keyPath: String, block: @escaping KVONotificationBlock) {
self.kvoController.observe(object, keyPath: keyPath, options:[.initial, .new], block:{(observer: Any?, object: Any, change: [String: Any]) in
DispatchQueue.main.async(execute: { () -> Void in
block(observer, change["old"], change["new"]);
})
})
}
func observe(_ object: Any?, keyPath: String, options: NSKeyValueObservingOptions, mainThread: Bool, block: @escaping KVONotificationBlock) {
self.kvoController.observe(object, keyPath: keyPath, options: options) { (observer, object, change: [String: Any]) -> Void in
if !mainThread || Thread.isMainThread == true {
block(observer, change["old"], change["new"]);
} else {
DispatchQueue.main.async(execute: { () -> Void in
block(observer, change["old"], change["new"]);
})
}
}
}
func removeObserve(_ object: Any?, path: String) {
self.kvoController.unobserve(object, keyPath: path);
}
func removeObserve(_ obj: Any?) {
self.kvoController.unobserve(obj);
}
func removeAll() {
self.kvoController.unobserveAll();
}
}
- kvoController是啥?
FBKVOController是Facebook开源的替代KVO的解决方案。它用block解决了以前使用KVO时代码散乱的缺点。可以利用CocoaPods,pod 'KVOController',到自己的工程中。 -
为什么能直接self.kvoController?
因为在FBKVOController中,有一个NSObject+FBKVOController扩展,kvoController是成员变量,所以NSObject能用.语法,拿到kvoController。
- 为啥要用FBKVOController?
大家都知道,kvo 全称 key-value observing,由 cocoa 框架提供的支持观察者模式的技术,结合 Objective-C 非常易用,在很多场合都可以有效地替换 NSNotificationCenter。但其也有一些致命的缺点,就是很容易导致引发 crash。譬如:只有addObserver,没有removeObserver。addObserver 和 removeObserver 必须配对出现,不然的话,等待你的就是crash。其调用的顺序:必须先添加观察者,然后处理业务,最后完成后移除观察者,释放掉观察者。removeObserver方法的调用,一般在dealloc()(OC)、deinit()(Swift)中。
而FBKVOController则是帮助我们更好的使用KVO。
FBKVOController生命周期跟观察者绑定,则观察者释放时,由FBKVOController生成的实例也被释放,从 _FBKVOSharedController 移除对应的观察者信息,避免发消息给已释放观察者导致的crash。 - 为啥要写这个扩展?
还不为了好用,省心省事呗,啊哈哈。在控制器中直接调用observe方法,传入对应的参数,在block回调中做该做的事情就可以。当然,也不需要在controller的dealloc(OC)或者deinit(Swift)方法中,调用removeObserver方法。例如下面的代码:
observe(scrollView, keyPath: #keyPath(UIScrollView.contentOffset)) { [weak self] (weakSelf, oldValue, newValue) in
// 处理事情