【iOS】手动实现KVO+Runtime

前言

KVO:简单的来说,就是观察者观察被观察对象属性的变化而发生相应的变化。实现的原理基于KVC与强大的Runtime机制。原理是什么?如何实现的?

系统实现步骤:

以下大概分为三步:

假设有类Person,它拥有一个年龄属性age。那么当Person类的对象第一次被观察的时候,系统会在运行期动态创建Person的派生类。我们如何知道?下面我们根据断点查看控制台即可。

[p addObserver:self forKeyPath:@"age" options:NSKeyValueObservingOptionNew context:nil];

上述代码执行前:


Jietu20171026-152507@2x.png

上述代码执行后:


Jietu20171026-152451@2x.png
  1. 通过以上我们可以得知在系统在运行期又动态的创建Person的派生类叫做NSKVONOtifying_Person.

  2. 在派生类NSKVONOtifying_Person中重写Person类的set方法,NSKVONOtifying_Person类在被重写的set方法中实现通知机制。此时类NSKVONOtifying_Person重写class方法,并且系统将所有原本指向类Person对象的isa指针指向类NSKVONOtifying_Person对象。

  3. 最后当Person类对象调用其set方法时,实质就是NSKVONOtifying_Person调用了重写的set方法,在set方法里用super关键字调用其父类的set方法,而后调用-(void)observeValueForKeyPath:ofObject:change:context:作出通知响应。

自定义实现

了解以上原理后,我们可以自己尝试实现

  • 创建NSObject的分类,自定义方法,在方法中创建中间派生类
  • 重写set方法
  • set方法中通知调用

下面贴上主代码:

NSObject+KVO.m中:
- (void)zz_addObserver:(NSObject *)observer forKeyPath:(NSString *)keyPath options:(NSKeyValueObservingOptions)options context:(nullable void *)context{
    self.zz_observer = observer;
    self.zz_keyPath = keyPath;
    
    NSString * className = [NSString stringWithFormat:@"ZZKVONotifying_%@",NSStringFromClass(self.class)];
    const  char *cla = className.UTF8String;
    Class subP =  objc_allocateClassPair([self class], cla, 0);
    class_addMethod(subP, @selector(setAge:), (IMP)setAge, "v@:@");
    objc_registerClassPair(subP);
    object_setClass(self, subP);
}

void setAge(id self , SEL _cmd,NSUInteger  age){
   
    NSString *keyPath =  objc_getAssociatedObject(self, (__bridge const void * _Nonnull)(observeKeyPathKey));
    NSObject *obj =  objc_getAssociatedObject(self, (__bridge const void * _Nonnull)(observeKey));
    [self willChangeValueForKey:keyPath];
    
    /*这里还应该调用父亲的[super setAge:age]方法*/  
    
    [self didChangeValueForKey:keyPath];
    
    [obj  observeValueForKeyPath:keyPath ofObject:self change:@{@"new":[NSNumber numberWithInteger:age]} context:nil];
}

通过以上,就实现了简单的KVO监听属性变化响应的功能。

PS

当然系统调用原比这要复杂的多。当添加观察者后,方法内部需要做很多的安全判断,如该对象是否实现了属性的set、get方法,如果 没有就抛出异常;是否已经存在该派生类对象,如果没有创建如果有就返回等等。另外,同一个对象的属性可以有多个观察者,所以内部必须要有一个集合去记录,当发生变化时,需要各个通知一一回调。

完结

文章中的代码只展示主要模块,如需要完整demo,请点我自行下载

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

推荐阅读更多精彩内容