KVO原理探秘

KVO的全称是Key-Value Observing,俗称“键值监听”,可以用于监听某个对象属性值的改变。

对于iOS开发者而言,相信大家都使用过KVO。但是对它的原理可能就不太清楚了。KVO是Runtime运行时机制的又一重大的应用,下面就对KVO的原理做一个详尽的阐述。

创建一个新的项目,然后在项目中创建一个ZPPerson类,这个类中有一个age属性。

ZPPerson类
age属性

然后在ViewController.m文件中创建一个person对象,并给它添加KVO监听。

创建person对象
给person对象添加KVO监听

当用户点击屏幕的时候,改变person对象的age属性的值。

改变age属性的值

这个时候系统就会调用监听方法了。

调用监听方法

下面阐述这个过程中KVO的实现原理:

  1. 在给这个person实例对象添加KVO监听之后,系统会利用Runtime机制动态地创建一个ZPPerson类的子类,名字就叫做"NSKVONotifying_ZPPerson"。然后系统会把这个person实例对象里面的isa指针由原来的指向ZPPerson类的class对象变为现在的指向NSKVONotifying_ZPPerson类的class对象;
  2. 新创建的子类"NSKVONotifying_ZPPerson"的class对象里面存储着isa指针、superclass指针、"setAge:"实例方法、"class"实例方法、"dealloc"实例方法以及"_isKVOA"实例方法等。其中系统会重写"setAge:"实例方法,重写后的该方法与它父类中的该方法的实现是不一样的,只不过方法的名称是一样的而已;
  3. 添加完KVO监听之后,当开发者调用"setAge:"实例方法来修改person对象的age属性的时候,根据上面所述,这个instance对象里面的isa指针已经由原来的指向ZPPerson类的class对象变为了现在的指向NSKVONotifying_ZPPerson类的class对象了,所以系统根据这个instance对象里面的isa指针找到的是NSKVONotifying_ZPPerson类的class对象,然后在这个class对象里面找到重写后的"setAge:"实例方法,最后再进行调用。这个重写的"setAge:"实例方法里面会调用Foundation框架里面的C语言函数"_NSSetIntValueAndNotify();",这个函数的实现里面首先会执行"[self willChangeValueForKey:@"age"];"代码,然后再执行"[super setAge:age];"代码,在执行这句代码的时候就会执行它的父类,也就是ZPPerson类里面的"setAge:"方法,从而真正更改这个属性的值,最后再执行"[self didChangeValueForKey:@"age"];"代码。"didChangeValueForKey:"这个方法里面会通知监听器"age"属性的值被改变了,即调用"- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context"监听方法,从而实现了对对象的某个属性进行监听的目的。

可以从下面的图中看出NSKVONotifying_ZPPerson类和ZPPerson类的关系:

NSKVONotifying_ZPPerson类和ZPPerson类的关系

除了上述的改变对象的属性值的时候会自动调用KVO的监听方法之外,想要在不改变对象属性值的时候也能自动调用KVO的监听方法,应该怎么做呢?

手动触发KVO:

在点击屏幕的时候调用手动触发KVO的方法:


调用手动触发KVO的方法

在方法中需要主动调用"willChangeValueForKey:"和 "didChangeValueForKey:"方法。

手动触发KVO方法

这样的话,在点击屏幕的时候就会手动触发KVO的监听方法了"- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context"。

image.png

以上就是对KVO原理的阐述。

Github Demo

补充:

修改对象里面的成员变量的值是不能够触发KVO的监听方法的,代码如下所示:

在ZPPerson类里添加一个_age成员变量:

_age成员变量

在ViewController.m文件中给age属性赋值:

给age属性赋值

点击屏幕的时候修改_age成员变量的值:

修改_age成员变量的值

运行之后可以知道通过这种方式改变对象的成员变量的值是无法自动触发KVO的监听方法的"- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context"。

修改对象的成员变量无法触发KVO监听方法的原因:

原因就在于对于已经被添加KVO监听的对象而言,instance对象的isa指针已经被改变为了指向系统新创建的子类的class对象了,然后在这个子类的class对象里面找到了(setAge:)实例方法,最后再进行调用。由之前的Demo可知,当调用新创建的子类的class对象中的属性的set实例方法的时候其实是在调用Foundation框架里面的C语言函数"_NSSetIntValueAndNotify",这个函数中会执行"didChangeValueForKey:"方法,此方法会触发KVO的监听方法。由此可知KVO的本质是调用set实例方法,即只有通过调用属性的set实例方法来修改属性的值才能触发KVO的监听方法,而修改对象的成员变量就没有调用属性的set方法,所以是不能够触发KVO的监听方法的。

如果想要在修改对象的成员变量的值的时候成功触发KVO的监听方法的话就要手动进行触发:

image.png

在手动触发方法的实现里面要调用"willChangeValueForKey:"和 "didChangeValueForKey:"方法:

image.png

运行之后就可以在控制台上看到成功调用了KVO的监听方法了:

控制台打印结果

Github Demo

”三人行,必有我师焉“, 欢迎各位批评指正。
如果您还觉得我写的不错的话请您点赞加关注,您的肯定是我前进的最大动力!
我是爱学习也爱您的树懒O(∩_∩)O

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

推荐阅读更多精彩内容