KVO梳理与使用

一、概述

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
  1. 当观察者不需要监听时,需要移除监听
- (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监听数组,也可以采用这种方式

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

友情链接更多精彩内容