KVO
KVO是Key-value observing,俗称键值监听(键值观察),可以用于监听某个属性值的改变。
当我们在调用 addObserver:forKeyPath时,系统会在运行时动态创建NSKVONotifying_A
类,同时将原来的类指针指向新创建的类。新创建的NSKVONotifying_A
类是原来类的子类,目的是重写setter方法,新的setter加入了通知所有观察对象的代码。
当修改instance对象的属性时,会调用Foundation的_NSSetXXXValueAndNotify函数
willChangeValueForKey:
父类原来的setter
didChangeValueForKey:
内部会触发监听器(Oberser)的监听方法( observeValueForKeyPath:ofObject:change:context:
那为什么只调用didChangeValueForKey:无法成功调用Observer的监听方法呢?
原因:为了在回调中能够提供监听的属性的旧值,在willChangeValueForKey:中,会通过KVC取得旧值。没有取得旧值,调用Observer的监听方法的条件不成立。
KVC
KVC的全称是Key-Value Coding,俗称“键值编码”,可以通过一个key来访问某个属性
常见的API有
- (void)setValue:(id)value forKeyPath:(NSString *)keyPath;
- (void)setValue:(id)value forKey:(NSString *)key;
- (id)valueForKeyPath:(NSString *)keyPath;
- (id)valueForKey:(NSString *)key;
setValue:forKey 底层原理
当调用setValue:forKey时,会先顺序查找该类对象的setKey和_setKey方法,找到了就调用方方法传递参数。没找到则查看accessInstanceVariablesDirectly
方法的返回值,返回值为NO则调用 setValue:forUndefinedKey:
并抛出异常 NSUnknownKeyException; 返回值为真则,顺序查找成员变量_key,_isKey,key,isKey,找到了直接赋值,没找到的话也是调用
setValue:forUndefinedKey: 并抛出异常
NSUnknownKeyException`;
valueForKey: 底层原理
当调用valueForKey:时,会先顺序查找该类对象的getKey、key、isKey、_key方法,找到了就调用方法获取返回值。
没找到则查看accessInstanceVariablesDirectly
方法的返回值,返回值为NO则调用valueForUndefinedKey:
并抛出异常NSUnknownKeyException
; 返回值为YES则,顺序查找成员变量_key,_isKey,key,isKey,找到了直接取值,没找到的话也是调用valueForUndefinedKey:
并抛出异常NSUnknownKeyException
面试题
- 什么是KVO
- KVO是Key-value observing 的缩写
- KVO是Objective-C对观察者设计模式的又一实现
- Apple使用了isa混写(isa-swizzling)来实现KVO
- isa混写技术在KVO中是怎么体现的?
当我们在调用 addObserver:forKeyPath时,系统会在运行时动态创建NSKVONotifying_A
类,同时将原来的类指针指向新创建的类。新创建的NSKVONotifying_A
类是原来类的子类,目的是重写setter方法,新的setter加入了通知所有观察对象的代码。
调用了NSSetIntValueAndNotify
实现如下:
- (void)setValue:(id)obj {
[self willChangeValueForKey:@"keyPath"];
[super setValue:obj];
[self didChangeValueForKey:@"keyPath"];
}
-
通过kvc设置value,KVO能否生效
能生效,kvc最终也是使用setter方法来赋值的。 -
通过成员变量直接赋值value,kvo能否生效?如果不能生效,如何让其生?
直接给成员变量赋值不走setter,因而KVO不会生效。要使其生效,除了改成访问属性之外,还可以在成员变量赋值的前后加上[self willChangeValueForKey:@"value"]
和[self didChangeValueForKey:@"value"]
手动触发使其生效。 -
KVC 的取值过程和原理是什么?