那些年刷过的题 - KVO

题目1. 在开发的过程中你使用过 KVO 么?如果使用过,它底层的实现原理是什么?

demo 肯定要第一时间放出来的啦~~~

答:这个答案正常都是有用过的,如果你没有用过。。。好吧看完这篇文章,赶紧去试下的。其实这个问题的重点本身不在于你是否使用过 KVO,但如果你说没有,估计你是会 gg 的了。问题的关键点还是回答出 KVO 的底层实现,实质上,这里还是 Runtime 的内容,关于 Runtime 的面试题目可以看上一篇文章的是消息派发,而 KVO 的具体实现请继续看下文。

KVO 的使用和原理

图片.png

图中 _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 说明:

那些年刷过的题 - KVO_第1张图片
图片.png

这里再提供一个 Swift 下使用的 KVO 时查看 isa 指针的方法:

那些年刷过的题 - KVO_第2张图片
图片.png

Swift 中 isa 指向的类名和 OC 中还是有所不同的:

NSKVONotifying_xxxx.SearchResultViewController // xxxx 为工程名,或者可以说是命名空间

因为 OC 中是没有命名空间的,所以和 Swift 自动生成的类的类名还是有所不同的。

附带问题 KVC

这部分内容直接复制粘贴下概念就行:

当调用 setValue 方法时,方法内部会做以下操作:

  1. 检查是否存在相应 key 的 set 方法,如果存在,就调用 set 方法
  2. 如果 set 方法不存在,就会查找与 key 相同名称并且带下划线的成员属性,如果有,则直接给成员属性赋值
  3. 如果没有找到 _key,就会查找相同名称的属性 key,如果有就直接赋值
  4. 如果还没找到,则调用 valueForUndefinedKey:setValue:forUndefinedKey: 方法。 这些方法的默认实现都是抛出异常。

你可能感兴趣的:(那些年刷过的题 - KVO)