iOS KVO实现细节验证

验证KVO

objective-c中的kvo是观察者模式的一种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");
    }

此处 NSKVONotifying_XXObc 就是系统内部实现的一个XXObc的子类。根据官方文档的解释,系统生成 NSKVONotifying_XXObc 类,然后通过交换原先类的isa指针来实现的KVO,怎么知道 NSKVONotifying_XXObc 类名的:通过调用下面方法

po object_getClassName([xx class])

控制台简单粗暴的打印。

此处为什么要比较,class 方法的IMP,上面apple的文档不是说了,不能依赖isa 指针来判断一个类,要通过class 所以,子类必然是重写过class方法的。

iOS KVO实现细节验证_第1张图片
调试过程

我们自己再创建一个子类,在addObserver之后修改一下xx的isa指针,看看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");
    }
    

object_setClass通过这个方法来改变isa指针

这下看到,XXObcobserveValueForKeyPath 方法不会再被调用了。

系统内部实现的子类,比如这里的NSKVONotifying_XXObc。 重写 class 方法,只是为了麻痹一下开发者,隐藏一下实现细节。也不影响通过 class 方法判断一下类,一举两得。

以上就是验证KVO的实现过程。

你可能感兴趣的:(iOS KVO实现细节验证)