KVC 的原理概述

作者: CoderHG | 来源:发表于2018-04-29 13:22 被阅读20次

    一、KVC 很简单

    KVC 很简单,每个人都会用,仅有的 API 如下:
    1、setValue: forKeyPath:
    2、setValue: forKey:
    3、valueForKeyPath:
    4、valueForKey:

    前两个是设置值(value),后两个是通过 key 或者 keyPath 获取对应的 value(值)。

    在接下来的介绍中,大家主要关注非常规用法。

    二、KVC 的实质

    通常所理解的 KVC 是:设置或者获取某个属性或者成员变量的值。但是问题来了,如果没有对应的属性获取成员变量,又会怎样呢?接下来分批介绍一下。

    2.1 常规用法

    有一个属性是这样定义的:

    // 名字
    @property (nonatomic, copy) NSString* name;
    

    我们可以直接通过属性名使用 KVC:

    KVCObject* kvcObj = [[KVCObject alloc] init];
    
    // 通过属性名设置具体的 value
    [kvcObj setValue:@"CoderHG" forKey:@"name"];
    // 通过属性获取具体的 value
    NSString* name = [kvcObj valueForKey:@"name"];
    NSLog(@"姓名: %@", name);
    

    以上的设置与获取 value 都会执行其属性的 setter 与 getter 方法。

    通过成员变量使用 KVC:

    KVCObject* kvcObj = [[KVCObject alloc] init];
    
    // 通过成员变量设置具体的 value
    [kvcObj setValue:@"CoderHG" forKey:@"_name"];
    // 通过成员变量获取具体的 value
    NSString* name = [kvcObj valueForKey:@"_name"];
    NSLog(@"姓名: %@", name);
    

    以上的设置与获取 value 都 会执行其属性的 setter 与 getter 方法。

    是的、在开发中只需要知道这两种常规的用法就足够了。但是对于程序员来说除了开发、还有一个名词叫 面试,如果面试的时候只知道上面的用法,那肯定是不够的。

    2.2 特别的 key 需要特别的处理

    这里的 特别的意思是,这个 key 不是对应的属性,也并非对应的成员变量,也可以称为 非常规。那么这种情况,KVC 又是如何处理的呢?看一下如下代码:

    KVCObject* kvcObj = [[KVCObject alloc] init];
    // 非常规设置具体的 value
    [kvcObj setValue:@"CoderHG" forKey:@"goddess"];
    // 非常规获取具体的 value
    NSString* name = [kvcObj valueForKey:@"goddess"];
    NSLog(@"姓名: %@", name);
    

    其中 goddess 既不是属性、也不似成员变量。这种情况,如果不做任何的处理,直接运行代码,那肯定会 crash 的。

    setValue: forKey: 的 crash 日志:

    *** Terminating app due to uncaught exception 'NSUnknownKeyException', reason: 
    '[<KVCObject 0x60800001eaa0> setValue:forUndefinedKey:]: 
    this class is not key value coding-compliant for the key goddess.'
    

    valueForKey: 的 crash 日志:

    *** Terminating app due to uncaught exception 'NSUnknownKeyException', reason: 
    '[<KVCObject 0x608000004c30> valueForUndefinedKey:]: 
    this class is not key value coding-compliant for the key goddess.'
    

    这种情况在 KVC 中就直接没救了么?不是的,其实在 crash 之前做了很多的查找工作的。

    setValue: forKey: 的查询顺序是这样的:

    • 1、依次查找是否实现了这些对象方法:setKey:、_setKey:。一旦找到其中的一个实现,则直接调用,都没有找到,则进入下一步查找。
    • 2、第一步未找到,调用 +accessInstanceVariablesDirectly 方法,如果返回为 NO,则直接 crash,返回为 YES,则会进入下一步。这个方法默认返回 YES。
    • 3、第二步返回 YES, 则会继续查找是否有如下的成员变量:_key、_isKey、key 与 isKey。找到则直接赋值,没有找到则直接 crash。

    注意:第一步查找的是 方法,第三部查找的是 成员变量

    valueForKey: 的查询顺序是这样的:

    • 1、按照顺序依次查找这些对象方法:getKey、key、isKey 与 _key。找到则执行,没有找到则进入下一步。
    • 2、调用 +accessInstanceVariablesDirectly 方法,如果返回为 NO,则直接 crash,返回为 YES,则会进入下一步。这个方法默认返回 YES。
      3、第二步返回 YES, 则会继续查找是否有如下的成员变量:_key、_isKey、key 与 isKey。找到则获取对应成员变量的值,没有找到则直接 crash。

    温馨提示: 关于第2、3步,两种情况是类似的。

    总结

    其实写本简书的重点是关注 非常规的情况,主要是记住其查找步骤即可。大概就是先查找方法、在查找成员变量。

    在写本简书的时候,我有一个试验 Demo # OC2Nature,可以作为一个参考。具体请看 KVC 目录。

    谢谢!

    相关文章

      网友评论

        本文标题:KVC 的原理概述

        本文链接:https://www.haomeiwen.com/subject/qfmolftx.html