说一说在Swift中使用kvo的一点小坑
kvo实现原理
kvo是基于runtime实现的
- 一个类的属性注册了
观察者
,在运行时,会动态的创建这个类的派生类(就是子类) - 键值观察通知依赖于NSObject 的两个方法:
willChangeValueForKey:
和didChangevlueForKey:
;在一个被观察属性发生改变之前,willChangeValueForKey:
一定会被调用,这就 会记录旧的值。而当改变发生后,didChangeValueForKey:
会被调用,继而observeValueForKey:ofObject:change:context
: 也会被调用。 - 子类中重写了 set与get方法,在set方法里:
- willChangeValueForKey
- 调用父类的 setValue for key这点很重要下面会细讲
- 然后调用didChangeValueForKey通知观察者
- 一个类的属性注册了
kvc 实现原理
上面说到 setValue for key 这是kvc
- 寻找该属性有没有setsetter方法?有,就直接赋值
- 寻找有没有该属性的成员属性?有,就直接赋值
- 寻找有没有该属性带下划线的成员属性?有,就直接赋值
kvo在swift触发问题
oc:一般直接给观察的那个属性赋新值 就会触发
这是因为 oc具有运行时机制,并且继承的NSObject类,而NSObject类实现了willChangeValueForKey
、didChangeValueForKey
能够在运行时派生一个子类正常执行。
swift: 1,给属性前面加上 关键字 dynamic ,2 使用setvalue for key 可以触发
swift本身屏蔽了运行时机制,没有运行时就没有派生类。没有派生类,就没有重写set get方法。没有重写set方法 ,就没有调用类本身的 setvalue for key 来 触发kvo通知观察者。so 要么加上关键字 dynamic
获取动态性、要么直接使用setvalue for key
来触发kvo。