说一说在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。

本文探讨了Swift中使用Key-Value Observing (KVO)遇到的问题,由于Swift的静态特性,直接赋值无法触发KVO。解决方法是在属性前添加`dynamic`关键字或使用`setValue(forKey:)`方法。KVO的实现基于Objective-C的runtime,通过创建派生类并重写setter来实现通知观察者。
746

被折叠的 条评论
为什么被折叠?



