<strong>验证<code>KVO</code></strong>
<code>objective-c</code>中的<code>kvo</code>是观察者模式的一种oc实现。
不管使用方式了,直接看看kvo是如何实现的。
苹果官方文档解释
简单粗暴的解释就是,在对一个class 进行kvo监听的时候,系统内部实现了一个子类,继承自 原先的类,然后 替换原先类的 isa指针,重写set方法,class 方法。
下面我们验证一下:
@import ObjectiveC.message;
@import ObjectiveC.runtime;
@interface XXObc : NSObject
@property (nonatomic, copy) NSString *ccc;
@end
@implementation XXObc
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context
{
NSLog(@"%@",keyPath);
}
@end
在 main 方法中实现
XXObc *xx = [XXObc new];
[xx addObserver:xx
forKeyPath:@"ccc"
options:NSKeyValueObservingOptionNew
context:nil];
xx.ccc = @"ccccccc";
IMP imp = class_getMethodImplementation(NSClassFromString(@"NSKVONotifying_XXObc"),
@selector(class));
IMP imp1 = class_getMethodImplementation([xx class], @selector(class));
if (imp != imp1) {
NSLog(@"11111");
}
此处 <code> NSKVONotifying_XXObc </code> 就是系统内部实现的一个<code>XXObc</code>的子类。根据官方文档的解释,系统生成<code> NSKVONotifying_XXObc </code>类,然后通过交换原先类的<code>isa</code>指针来实现的KVO,怎么知道<code> NSKVONotifying_XXObc </code>类名的:通过调用下面方法
po object_getClassName([xx class])
控制台简单粗暴的打印。
此处为什么要比较,<code>class</code> 方法的<code>IMP</code>,上面apple的文档不是说了,不能依赖<code>isa</code> 指针来判断一个类,要通过<code>class</code> 所以,子类必然是重写过<code>class</code>方法的。
我们自己再创建一个子类,在<code>addObserver</code>之后修改一下xx的<code>isa</code>指针,看看kvo有没有效果。
添加如下子类。
@interface XXXChildObj : XXObc
@end
@implementation XXXChildObj
- (void)setCcc:(NSString *)ccc {
[super setCcc:ccc];
}
@end
调试代码稍微修改
XXObc *xx = [XXObc new];
[xx addObserver:xx
forKeyPath:@"ccc"
options:NSKeyValueObservingOptionNew
context:nil];
object_setClass(xx,
[XXXChildObj class]);
xx.ccc = @"ccccccc";
IMP imp = class_getMethodImplementation(NSClassFromString(@"NSKVONotifying_XXObc"),
@selector(class));
IMP imp1 = class_getMethodImplementation([XXXChildObj class], @selector(class));
if (imp != imp1) {
NSLog(@"11111");
}
<code>object_setClass</code>通过这个方法来改变isa指针
这下看到,<code>XXObc</code> 的 <code>observeValueForKeyPath</code> 方法不会再被调用了。
系统内部实现的子类,比如这里的<code>NSKVONotifying_XXObc</code>。 重写 class 方法,只是为了麻痹一下开发者,隐藏一下实现细节。也不影响通过 class 方法判断一下类,一举两得。
以上就是验证KVO的实现过程。