题目1. 在开发的过程中你使用过 KVO 么?如果使用过,它底层的实现原理是什么?
demo 肯定要第一时间放出来的啦~~~
答:这个答案正常都是有用过的,如果你没有用过。。。好吧看完这篇文章,赶紧去试下的。其实这个问题的重点本身不在于你是否使用过 KVO,但如果你说没有,估计你是会 gg 的了。问题的关键点还是回答出 KVO 的底层实现,实质上,这里还是 Runtime 的内容,关于 Runtime 的面试题目可以看上一篇文章的是消息派发,而 KVO 的具体实现请继续看下文。
KVO 的使用和原理
图中 _cat 注册监听后,可以看到 _cat 的 isa 指针指向的是 NSKVONotifying_Cat,这里就可以看出,KVO 中,实质是通过 runtime 重新指向一个 Cat 的子类 NSKVONotifying_Cat ,里面再重写对应的 setter 方法,大概就是这个样子:
#import "NSKVONotifying_Cat.h"
@implementation NSKVONotifying_Cat
- (void)setColor:(UIColor *)color {
[self willChangeValueForKey:@"color"];
[self didChangeValueForKey:@"color"];
[super setColor:color];
}
@end
刚看这里的代码时,应该会有个疑问,这里为什么要用 willChangeValueForKey 和 didChangeValueForKey,这里直接看它的 api 说明:
这里再提供一个 Swift 下使用的 KVO 时查看 isa 指针的方法:
Swift 中 isa 指向的类名和 OC 中还是有所不同的:
NSKVONotifying_xxxx.SearchResultViewController // xxxx 为工程名,或者可以说是命名空间
因为 OC 中是没有命名空间的,所以和 Swift 自动生成的类的类名还是有所不同的。
附带问题 KVC
这部分内容直接复制粘贴下概念就行:
当调用 setValue 方法时,方法内部会做以下操作:
- 检查是否存在相应 key 的 set 方法,如果存在,就调用 set 方法
- 如果 set 方法不存在,就会查找与 key 相同名称并且带下划线的成员属性,如果有,则直接给成员属性赋值
- 如果没有找到 _key,就会查找相同名称的属性 key,如果有就直接赋值
- 如果还没找到,则调用 valueForUndefinedKey: 和 setValue:forUndefinedKey: 方法。 这些方法的默认实现都是抛出异常。