KVC
什么是KVC
是一种键值编码机制(key-value),通过NSKeyValueCoding协议来间接访问成员变量
它会破坏面向对象编程思想,上面的key是没有任何限制的,当我们知道一个类内部的某个私有成员变量名称,就可以通过key进行设置和访问
作用
可以随意修改一个对象的属性或者成员变量,因为key没有限制,私有也可以
怎么用
赋值: 设置某一对象当中和key同名或者相似名字的实例变量的值
- (void)setValue:(id)value forKey:(NSString *)key;
- (void)setValue:(id)value forKeyPath:(NSString *)key; 通过点可层层赋值
取值: 获取和key同名或者相似名字的实例变量的值
- (id)valueForKey:(NSString *)key;
- (id)valueForKeyPath:(NSString *)key;
系统内部实现流程
`赋值过程`
先找相关方法set<Key> _set<Key> setIs<Key>
若没有相关方法,判断是否可以直接方法成员变量accessInstanceVariablesDirectly == YES ?
YES:继续找相关变量 _<Key> _is<Key> <Key> is<Key>赋值,若都不存在,则调forUndefinedKey,抛出异常
NO:执行setValue: forUndefinedKey 系统抛出异常:未定义的key
取值过程
先找相关方法get<Key> <Key> <countOfKey> && objectInKeyAtIndex
若没有相关方法,判断是否可以直接方法成员变量accessInstanceVariablesDirectly == YES ?
YES: 继续找相关变量 _<Key> _is<Key> <Key> is<Key>赋值,若都不存在,则调forUndefinedKey,抛出异常
NO: 执行setValue: forUndefinedKey 系统抛出异常:未定义的key
其他
赋值为空: setNilValueForKey
key值不存在: setValue: forUndefinedKey
validateValue方法的工作原理: 属性值正确性验证,用来检查set的值是否正确、为不正确的值做一个替换值或者拒绝设置新值并返回错误原因
关于setValue的调用流程
关于valueForKey的调用流程
KVO - Key-value observing
什么是KVO
- 是OC对于观察者设计模式的一种实现,每次当被观察对象的某个属性值发生改变时,注册的观察者便能获得通知。
- KVO是NSObject的一个分类
KVO怎么用
1.添加观察
[对象 addObserver:观察者 forKeyPath:被监听的属性 options:值如何变化 context:可以传参];
2.观察回调
[observeValueForKeyPath:哪个属性被改了 ofObject:哪个对象的属性被改了 change:值 context:参数]
3.移除removeObserver
KVO是如何实现的
Apple使用isa混写技术(isa-swizzling)来实现KVO,本质是重写了setter方法
当调用了addObserver: forKeyPath: 方法后,系统会在运行时,为这个类动态的创建了一个子类(NSKVONotifying_A),改写isa指向(将原来类A的isa指针指向NSKVONotifying_A这个类),同时重写setter方法,来实现KVO机制
通过下图可以看到,在调用[addObserver forKeyPath]之前,类还是MObject,在调用方法之后,类就变成了NSKVONotifying_MObject,其中isa指针的指向也发生了修改
KVO原理是重写setter方法
NSKVONotifying_A类实际是类A的子类,之所以继承是为了重写类A的setter方法, NSKVONotifying_A通过对setter方法的重写,达到了可以通知所有观察者的目的
为什么更改setter就可以实现KVO的监听呢
NSKVONotifying_A重写了原来类的setter方法
具体实现是下面两行代码
- (void)willChangeValueForKey:(NSString *)key;
- (void)didChangeValueForKey:(NSString *)key;
如图所示
首先调用willChangeValueForKey方法
然后调用父类实现,也就是原来类的setter实现
然后调用didChangeValueForKey,这个方法会触发observeValueForKeyPath这个KVO回调,来通知观察者value发生了变化
我们做哪些操作可以触发KVO(Demo百度网盘)
因为本质是重写setter,所以肯定是要和setter相关才可以
1.setter相关:点方法和KVC赋值(内部是setter赋值)
点方法: self.p.name = @"111";
KVC赋值: [self.p setValue:@"111" forKey:@"name"];
2.自己写代码手动调用
KVO内部也是在setter方法中调用这两句代码
在值改变前调用方法[self willChangeValueForKey:@"value"];
在值改变后调用方法[self didChangeValueForKey:@"value"];
哪些操作不响应KVO
若通过方法调用直接给成员变量赋值,不会被监听
[obj increase];
- (void)increase
{
_value += 1;
}
总结
- 使用setter方法改变值,KVO回调可以生效
- 使用setValue: forKey: 改变值,KVO回调可以生效
- 成员变量直接赋值KVO无法生效,必须手动添加KVO两个方法,KVO回调才会生效