KVOController
是目前OC上用的最多的KVO的第三方库,facebook
出品。有以下特点:
- 提供
block
方式和selector
方式,不需要实现observeValueForKeyPath:ofObject:change:context:
- 不需要关心
context
,keyPath
等参数来判断是哪个观察者的事件 - 在某些情况下,不需要手动移除观察者
本文主要探讨以下内容:
1. 类之间的引用关系
2. 什么情况可以自动移除观察者,不需要开发者关心
3. KVOControllerNonRetaining
4. 实际开发中怎么用更方便
类之间的引用关系
如下图所示:
- 调用者持有
KVOController
,并调用KVOController
中的observe:keyPath:options:block:
方法注册观察者,并在block中处理自己的业务 -
_FBKVOInfo
记录调用者的信息,并作为context
上下文参数传递。当收到系统回调时取出该上下文,并回调给调用者 -
_FBKVOSharedController
是一个单例,真正调用系统addObserver
的观察者 -
NSObject+FBKVOController
分类中提供两个属性,KVOController
和KVOControllerNonRetaining
4.1 这两个属性都是strong
引用,即调用者会强引用KVOController
4.2KVOController
表示被观察者也是强引用的
4.3KVOControllerNonRetaining
表示被观察者是弱引用
理想情况下的自动解除KVO
从上图可以看到,KVOController
唯一被强引用的地方就是self(observer)
。所以,只要self
对象被释放,KVOController
随后就会被释放。在KVOController
的dealloc
方法中,调用了unobserveAll
,从而实现了自动移除所有观察者的功能。
但现实总是残酷的,总有可能不小心导致self
对象释放不了,如:
-
block
中使用了self
,导致block
强引用self
,此时调用者就无法自动释放了 - 被观察者是
self
,此时KVOController
的_objectInfosMap
属性中的key
即为self
,且是强引用的,所以调用者也无法释放
KVOControllerNonRetaining
在上面的被观察者是self
的情况时,可能有人注意到了在 NSObject+FBKVOController
分类中提供了一个属性:KVOControllerNonRetaining
。是不是可以用这个属性就可以做到自动移除观察者了呢?
答案是否定的,这种情况依然无法移除观察者,而且会造成崩溃。
这是因为:
-
KVOControllerNonRetaining
表示_objectInfosMap
属性中的key
为弱引用。 - 此时
observer == observed == self
- 当
self
被释放时,KVOControllerNonRetaining
会执行dealloc
中的unobserveAll
,由于_objectInfosMap
属性中的key
为弱引用,NSMapTable
会自动清除该键值对,所以永远移除不了观察者。而self
对象,由于还依然存在着注册的观察者(_FBKVOSharedController)
,最终导致崩溃。
回归实际
从我个人实际的开发角度而言,self
通常既是我的观察者,也是我的被观察者,因为就算我观察的不是类本身的属性,也通常是我类的属性的属性。而此时我通常喜欢使用被观察者是self
的原因是OC
提供的Keypath
方式,通过FBKVOKeyPath
宏可以方便的进行点.
操作,如FBKVOKeyPath(self.clock.date)
。
所以FBKVOController
对我而言,常用的方式是:
- 使用分类中的
KVOController
属性 - 调用带
block
的方法 - 手动移除观察者