知识点总结中,如有幸被您观看,更有可能看到不足,期待指出交流
前言
继续学习中.....
KVC的底层实现
当一个对象调用serValue方法时,方法内部会做下面的操作:
- 检查是否存在相同的key的set方法,如果不存在,就调用set方法
- 如果set方法不存在,就会查找与key相同名称并且带下划线的成员属性.如果有,则直接给成员属性赋值.
- 如果没有找到_key,就会查找相同名称的属性key,如果有就直接赋值
- 如果还没有找到,则调用
valueForUndefinedKey:
setValue:forUnderfinedKey:
这些方法的默认实现都是抛出异常,我们可以根据需要重写他们.
KVO的底层实现
- KVO基于runtime机制实现
- 使用 isa 混写 (isa-swizzling), 当一个对象(假设person对象,person的类是MYperson) 的属性值(假设person的age)发生改变时,系统会自动生成一个类,继承自MYPerson:NSKVONotifying_MYPerson,在这个类的setAge方法里面,调用
[super setAge:age]
[self willChangeValueForKey:@"age"]
[self didChangeValueForKey:@"age"]
下面这俩个方法内部会主动调用监听者内部的
-(void)observeValueForKeyPath
- 想要看到 NSKVONotifying_MYPerson 很简单, 在self.person.age = 20; 这里打断点,在调试区域就能看到
_person->NSObject-isa=(Class)MYPerson
由此可见,person类型已经由MYperson被改变成NSKVONotifing_MYperson
什么是KVO和KVC
KVC: 键值编码,使用字符串直接访问对象的属性
KVO: 键值观察机制,它提供了观察某一属性变化的方法
KVO的缺陷
KVO是一个对象能够观察另一个对象的属性的值,并且能够发现值的变化.block和代理,两种模式更加适合一个Controller与任何其他的对象进行通信.而KVO更加适合任何类型的对象监听另外一个任意对象(这里也可以是Controller,但一般不是Controller).这个一个对象与另外一个对象保持同步的一种方法,即单另外一种对象的状态翻身改变时,观察对象马上做出反应.它只能用来对属性做出反应,而不会用来对方法或者动作做出反应.
优点:
- 能够提供一种简单的方法实现俩个对象间的同步,列如model和View之间的同步
- 能够对非我们创建的对象,即内部对象的状态改变做出相应,而且不需要改变内部对象(SKD对象)的实现
- 能够提供观察的属性的最新值以及先前值
- 用 key paths 来观察属性, 因此也可以观察嵌套对象
- 完成了对观察对象的都想,因为不需要额外的代码来允许键值被观察
确定: - 我们观察的属性必须使用string来定义.因此编译器不会出现警告以及检查
- 对属性重构导致我们的观察代码不可再用
- 复杂的"IF"语句要求对象正在观察多个值,这是因为所有的观察代码通过一个方法来指向
- 当释放观察者时需要移除观察者