iOS KVO实现细节验证

<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的实现过程。

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

推荐阅读更多精彩内容

  • 转至元数据结尾创建: 董潇伟,最新修改于: 十二月 23, 2016 转至元数据起始第一章:isa和Class一....
    40c0490e5268阅读 1,767评论 0 9
  • 一、概述 KVO,即:Key-Value Observing,它提供一种机制,当指定的对象的属性被修改后,则其观察...
    DeerRun阅读 10,112评论 11 33
  • 上半年有段时间做了一个项目,项目中聊天界面用到了音频播放,涉及到进度条,当时做android时候处理的不太好,由于...
    DaZenD阅读 3,038评论 0 26
  • 本文分为2个部分:概念与应用。概念部分旨在剖析 KVO 这一设计模式的实现原理;应用部分通过创建的项目,以说明 K...
    啊左阅读 57,941评论 107 438
  • iOS--KVO的实现原理与具体应用 长时间不用容易忘,这篇文章挺好的.转载自看本文分为2个部分:概念与应用。概念...
    超_iOS阅读 1,449评论 0 17