一、概述
KVO全称KeyValueObserving,是苹果提供的一套事件通知机制。允许对象监听另一个对象特定属性的改变,并在改变时接收到事件。对于基本数据类型是观察值的变化,但对于指针类型(OC中对象都是指针),是观察指针的地址是否变化。
二、KVO使用三步骤
1.注册观察者
- (void)addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(nullable void *)context;
2.监听方法,KVO会回调该方法方法来通知观察者。
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context
- 当观察者不需要监听时,需要移除监听
- (void)removeObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath context:(nullable void *)context
三、解读
1.注册方法
-
observer
:观察者对象,这个对象必须实现observeValueForKeyPath:ofObject:change:context:
方法,以响应属性的修改通知。 -
keyPath
:被监听的属性。这个值不能为nil。 -
options
:监听选项,参数是一个枚举类型。
options | 描述 |
---|---|
NSKeyValueObservingOptionNew |
提供属性的新值 |
NSKeyValueObservingOptionOld |
提供属性的旧值 |
NSKeyValueObservingOptionInitial |
观察最初的值(在注册观察服务时会调用一次触发) |
NSKeyValueObservingOptionPrior |
分别在值修改前后触发方法(即一次修改有两次触发) |
-
context
:任意的额外数据,我们可以将这些数据作为上下文数据,它会传递给观察者对象的observeValueForKeyPath:ofObject:change:context:方法。
在调用addObserver方法后,KVO并不会对观察者进行强引用,所以需要注意观察者的生命周期,否则会导致观察者被释放带来的Crash。
2. 监听方法
-
keyPath
:即被观察的属性,与参数object相关。 -
object
:keyPath所属的对象。 -
change
:这是一个字典,它包含了属性被修改的一些信息。这个字典中包含的值和我们在添加观察者时设置的options参数相呼应。 -
context
:这个值即是添加观察者时传递的数据。
3. 调用方式
1.直接调用set方法,或者通过属性的点语法间接调用
2.使用KVC的setValue:forKey:
方法
3.需要注意:
对于容器类,例如NSMutableArray在调用 addObject或removeObject 系列方法时,不会触发KVO。为了实现容器类的 KVO,官方为我们提供了如下方法:
- (NSMutableArray *)mutableArrayValueForKey:(NSString *)key
通过对该方法返回的数组进行增删操作可以触发KVO。
为什么会这样呢?
1.开头笔者提到过,对于指针对象,KVO本质上是监听指针的地址是否变化。
数组在调用它的 addObject、removeObject 系列方法时,虽然数组内部发生了变化,但是数组的地址并没有改变。所以,不会触发KVO。
2.调用mutableArrayValueForKey
这个方法,会返回一个代理对象。(你可以理解为被观察数组的深拷贝的对象)。至于深拷贝,浅拷贝的问题,笔者准备单独展开一期进行论述。
3.当对这个代理对象进行增删等操作后,会立刻对这个代理对象进行一次深拷贝,并将源数组的指针指向这个新拷贝出来的数组。(可能有点绕,仔细缕一下)。这样被观察的数组指针地址就发生了变化,从而触发KVO。
最后提一下,RAC的监听机制和KVO是一样的。所以,用RAC监听数组,也可以采用这种方式