KVO/KVC的底层原理和使用

KVO(Key-Value-Observe 键值观察)的原理并不复杂,但是涉及到isa指针superClass指针以及runtimeOC消息分发的知识,非常容易发散,所以一直是面试热点。

KVC键值编码 是Key Value Coding 的简称,cocoa的标准组成部分,是一种可以直接通过字符串的名字(Key)来访问类属性的机制,而不是通过调用Setter方法、Getter方法进行访问。

面试

(文末回答,也请评论你遇到的面试问题,共同进步。)

如何手动实现KVO? 如何解除KVO?KVO优缺点?
KVC是什么原理?能够使用KVO监听吗?
KVC赋值异常处理

原理

被添加监听的类Person,会在运行时动态创建一个该类的子类NSKVONotifying_ Person(superClass指向Person, isa指向自己的元类)。runtime会动态更改Person类的实例对象person的isa指向。当执行person.ageset方法时,会根据isa找到person 的类对象,找到setAge:方法(setAge:会执行Foundation框架的一个C方法_NSSetIntValueAndNotify_NSSetIntValueAndNotify的实现伪代码如下:

{
     [self.person1 willChangeValueForKey:@"age"];
     [super setAge:10];
     [self.person1 didChangeValueForKey:@"age"];
}

didChangeValueForKey:会触发监听方法 [observer observeValueForKeyPath:key ofObject:self change:change context:NULL]; 。

[super setAge:10];会执行父类的setAge方法。

添加监听的实例对象结构图
未添加监听的实例对象结构图

用途:

主要用于监听属性值的变化。可用于MVVMviewModelView的交互。(请在评论区继续ADD...)

扩展:

动态创建类

动态创建类参数:父类,类名,额外的内存空间

Class objc_allocateClassPair(Class superclass, const char *name, size_t extraBytes)

如何更改isa指向和isa指针的结构?

修改设置isa指向:

 object_setClass(id obj, Class cls)

isa和superClass指向:

  • 实例对象的isa 指针类对象,类对象的isa指针指向metaClass,metaClass的isa指针指向基类NSObject.

  • 实例对象没有superClass指针,类对象的superClass指向父类对象,一直到基类的类对象[NSObject class], NSObject的类对象指向nil。

  • metaClass对象的superClass指向父类的metaClass对象,一直到基类的metaClass对象, NSObject的metaClass对象指向类对象[NSObject class]。

isa和superClass指向

面试参考答案

如何手动实现KVO?

1、手动创建子类,并修改实例对象isa指向:
2、重写set方法,+class方法
3、重写didChangeValueForKey:

如何解除KVO?

重写didChangeValueForKey:

KVC/KVO的优缺点

  • KVC优点:没有property的变量(私有)也能通过KVC进行设置,json或者简化代码(多级属性)或者json转model 简化代码
  • KVC缺点:如果key只写错,编写的时候不会报错,但是运行的时候会报错

KVO优点

  • 能够提供一种简单的方法实现两个对象的同步;
  • 能够对内部对象的状态改变作出响应,而且不需要改变内部对象的实现;
  • 能够提供被观察者属性的最新值和之前的值;
  • 使用key Path来观察属性,因此可以观察嵌套对象;
  • 完成了对观察对象的抽象,因为不需要额外的代码来允许观察者被观察。

KVO缺点

  • KVO只能检测类中的属性,并且属性名都是通过NSString来查找,编译器不会补全(编译时不会出现警告),容易写错;
  • 对属性重构,将导致观察代码不可用;
  • 复杂的 “if” 语句要求对象正在观察多个值,是因为所有的观察代码通过一个方法来指向;

KVC能够使用KVO监听吗

KVC的API如下所示:

- (void)setValue:(id)value forKeyPath:(NSString *)keyPath;
- (void)setValue:(id)value forKey:(NSString *)key;
- (id)valueForKeyPath:(NSString *)keyPath;
- (id)valueForKey:(NSString *)key; 

KVC访问变量的流程如下图所示:


setValueforkey的执行流程

setValueforkey首先调用的是setKey方法,OC属性声明后或自动生成set 、 get 方法和_key的局部变量,所以默认是可以被KVO监听到的。
但是如果属性被readOnly修饰就不会自动生成set方法, 但是如果用KVC的话仍然可以修改被readOnly修饰的值。而且能够出发KVO监听,证明了下面的流程图:寻找_key的局部变量直接修改,并且主动调用willChangeValueForKey 和didChangeValueForKey, 触发KVO监听。(思考一下这是readOnly的漏洞吗?怎么解决呢?评论区见)

getValueforkey的执行流程

KVC赋值异常处理

- (void)setNilValueForKey:(NSString *)key
{
    NSLog(@"这里处理当赋值为nil时,出现异常");
}
- (void)setValue:(id)value forUndefinedKey:(NSString *)key
{
    NSLog(@"key没有定义的时候,可以在这里处理");
}

如有错误或者新的见解欢迎在评论区约谈...

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 222,252评论 6 516
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 94,886评论 3 399
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 168,814评论 0 361
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 59,869评论 1 299
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 68,888评论 6 398
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 52,475评论 1 312
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 41,010评论 3 422
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,924评论 0 277
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 46,469评论 1 319
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 38,552评论 3 342
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,680评论 1 353
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 36,362评论 5 351
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 42,037评论 3 335
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,519评论 0 25
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,621评论 1 274
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 49,099评论 3 378
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,691评论 2 361

推荐阅读更多精彩内容