前言
KVO:简单的来说,就是观察者观察被观察对象属性的变化而发生相应的变化。实现的原理基于KVC与强大的Runtime机制。原理是什么?如何实现的?
系统实现步骤:
以下大概分为三步:
假设有类Person,它拥有一个年龄属性age。那么当Person类的对象第一次被观察的时候,系统会在运行期动态创建Person的派生类。我们如何知道?下面我们根据断点查看控制台即可。
[p addObserver:self forKeyPath:@"age" options:NSKeyValueObservingOptionNew context:nil];
上述代码执行前:
上述代码执行后:
通过以上我们可以得知在系统在运行期又动态的创建Person的派生类叫做NSKVONOtifying_Person.
在派生类NSKVONOtifying_Person中重写Person类的set方法,NSKVONOtifying_Person类在被重写的set方法中实现通知机制。此时类NSKVONOtifying_Person重写class方法,并且系统将所有原本指向类Person对象的isa指针指向类NSKVONOtifying_Person对象。
最后当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,请点我自行下载。