KVC 的原理概述

一、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: 
'[ setValue:forUndefinedKey:]: 
this class is not key value coding-compliant for the key goddess.'

valueForKey: 的 crash 日志:

*** Terminating app due to uncaught exception 'NSUnknownKeyException', reason: 
'[ 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 的原理概述)