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

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容