KVO的原理 && KVC

KVO

kvo 即键值观察 观察一个对象的属性的变化,并在改变时接收到事件
kvo是如何监听属性变化的呢

[self.person1 addObserver:self forKeyPath:@"age" options:options context:@"123"];

首先要注册监听

self.person1.age = 10

当age属性变化的时候 会触发监听 收到回调

- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context

当添加addObserver:forKeyPath:options:context
runtime会生成一个继承与原类的中间类 NSKVONotifying_xxx
并动态修改实例对象的isa 指向这个中间类, 重写中间类的class 方法,返回原类的Class, 如果通过runtime去获取被监听实例对象的类对象,拿到的就是中间类 object_getClass() ,如果用 [person Class]获取类对象获取到的就是原类的类对象
被监听的实例对线的中间类会重写keyset方法 ,调用NSSetxxxValueAndNotify函数, 执行willChangeValueForKeysuper setter方法修改属性的值、didChangeValueForKey
didChangeValueForKey方法 内部会触发监听observeValueForKeyPath:ofObject:change:context:
中间类还会重写 dealloc方法 销毁

如何关闭系统的kvo呢?
重写 +(BOOL)automaticallyNotifiesObserversForKey:(NSString *)key 返回NO 即可

如何 手动触发kvo
手动调用willChangeValueForKey:和didChangeValueForKey:

只有执行set方法才会触发kvo,所以修改成员变量或者只修改变属性的值是不会触发kvo的,kvc例外, kvc对kvo有特殊的处理

KVC

accessInstanceVariablesDirectly 这个方法只针对所有的getter实现都没有的情况,是否可以去访问实例变量

赋值过程
按照setKey:、_setKey:顺序查找方法
如果没找到方法,并且accessInstanceVariablesDirectly返回YES,就开始查询成员变量
按照_key、_isKey、key、isKey的顺序查找成员变量

WeChata86c1b05cdd453781b6674771b11194c.png

取值过程
按照getKey、key、isKey、_key的顺序查找
如果accessInstanceVariablesDirectly返回YES,就直接查询成员变量,否则直接崩溃
按照_key、_isKey、key、isKey的顺序查找成员变量

WeChatadaa7ffca4edf48df622b3821f5864d9.png
setObject: forkey 和 setValue: forKey 区别

1、setObject: forkey: 中 object 是不能够为 nil 的,不然会报错。
setValue: forKey: 中 value 能够为 nil,但是当 value 为 nil 的时候,会自动调用 removeObject: forKey:方法。
2、setValue: forKey:中 key 的参数只能够是 NSString 类型,而
setObject: forkey: 中的 key 可以是任何类型。
3、setObject: forKey: 方法是 NSMutabledictionary 特有的,setValue: forKey:方法是 KVC(键-值编码)的主要方法

你可能感兴趣的:(KVO的原理 && KVC)