美文网首页Objective-C
【iOS】手动实现KVO+Runtime

【iOS】手动实现KVO+Runtime

作者: Colleny_Z | 来源:发表于2017-10-26 19:06 被阅读30次

前言

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,请点我自行下载

相关文章

  • 【iOS】手动实现KVO+Runtime

    前言 KVO:简单的来说,就是观察者观察被观察对象属性的变化而发生相应的变化。实现的原理基于KVC与强大的Runt...

  • 手动实现KVO+Runtime

    前言 KVO:观察者观察被观察者的属性的变化实现原理:KVC+runtime 自动实现 大概分为三步:假设有类pe...

  • iOS中实现多线程的几种方案,各自有什么特点?

    iOS中实现多线程的几种方案,各自有什么特点? NSThread 面向对象的,需要程序员手动创建线程,但不需要手动...

  • iOS-手动实现KVO

    我的Github地址 : Jerry4me, 本文章的demo链接 : JRCustomKVODemo 前言 KV...

  • iOS轻松实现手动布局

    画UI是每一个iOS开发工程师的必经之路.想必这个方法大家都不陌生CGRectMake(CGFloat x, CG...

  • 【KVO之手动触发】

    iOS 中的kvo分为自动监听和手动触发两种方式,自动监听很简单,不在此分析,这里主要说一下手动触发的实现,面试...

  • UIWebView实现自动登陆

    IOS中UIWebView不会自动存储、读取Cookie.要实现UIWebView的自动登陆,需要手动存储、读取C...

  • iOS KVO的手动实现过程

    如何手动触发一个value的KVO 手动实现 willChangeValueForKey 和 didChangeV...

  • iOS 利用runtime手动实现KVO

    KVO原理:调用监听对象属性的方法,动态创建一个继承自该对象所属类的子类,然后重写该属性的setter方法,在se...

  • iOS 高级研发

    iOS中使用OpenGL 实现增高功能 功能效果 demo示例 功能分析 功能:渲染一张传入的图片 -> 手动选择...

网友评论

    本文标题:【iOS】手动实现KVO+Runtime

    本文链接:https://www.haomeiwen.com/subject/xrakpxtx.html