KVOController

KVOController是目前OC上用的最多的KVO的第三方库,facebook出品。有以下特点:

  • 提供block方式和selector方式,不需要实现observeValueForKeyPath:ofObject:change:context:
  • 不需要关心contextkeyPath等参数来判断是哪个观察者的事件
  • 在某些情况下,不需要手动移除观察者

本文主要探讨以下内容:

1. 类之间的引用关系
2. 什么情况可以自动移除观察者,不需要开发者关心
3. KVOControllerNonRetaining
4. 实际开发中怎么用更方便

类之间的引用关系

如下图所示:

  1. 调用者持有KVOController,并调用KVOController中的observe:keyPath:options:block:方法注册观察者,并在block中处理自己的业务
  2. _FBKVOInfo记录调用者的信息,并作为context上下文参数传递。当收到系统回调时取出该上下文,并回调给调用者
  3. _FBKVOSharedController是一个单例,真正调用系统addObserver的观察者
  4. NSObject+FBKVOController分类中提供两个属性,KVOControllerKVOControllerNonRetaining
    4.1 这两个属性都是strong引用,即调用者会强引用KVOController
    4.2 KVOController表示观察者也是强引用的
    4.3 KVOControllerNonRetaining表示观察者是弱引用
类引用关系.png

理想情况下的自动解除KVO

从上图可以看到,KVOController唯一被强引用的地方就是self(observer)。所以,只要self对象被释放,KVOController随后就会被释放。在KVOControllerdealloc方法中,调用了unobserveAll,从而实现了自动移除所有观察者的功能。

现实总是残酷的,总有可能不小心导致self对象释放不了,如:

  1. block中使用了self,导致block强引用self,此时调用者就无法自动释放了
  2. 被观察者是self,此时KVOController_objectInfosMap属性中的key即为self,且是强引用的,所以调用者也无法释放

KVOControllerNonRetaining

在上面的被观察者是self的情况时,可能有人注意到了在 NSObject+FBKVOController分类中提供了一个属性:KVOControllerNonRetaining。是不是可以用这个属性就可以做到自动移除观察者了呢?
答案是否定的,这种情况依然无法移除观察者,而且会造成崩溃。
这是因为:

  1. KVOControllerNonRetaining表示_objectInfosMap属性中的key为弱引用。
  2. 此时observer == observed == self
  3. self被释放时,KVOControllerNonRetaining会执行dealloc中的unobserveAll,由于_objectInfosMap属性中的key为弱引用,NSMapTable会自动清除该键值对,所以永远移除不了观察者。而self对象,由于还依然存在着注册的观察者(_FBKVOSharedController),最终导致崩溃。

回归实际

从我个人实际的开发角度而言,self通常既是我的观察者,也是我的被观察者,因为就算我观察的不是类本身的属性,也通常是我类的属性的属性。而此时我通常喜欢使用被观察者是self的原因是OC提供的Keypath方式,通过FBKVOKeyPath宏可以方便的进行点.操作,如FBKVOKeyPath(self.clock.date)
所以FBKVOController对我而言,常用的方式是:

  1. 使用分类中的KVOController属性
  2. 调用带block的方法
  3. 手动移除观察者

你可能感兴趣的:(KVOController)