一、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 目录。