1. KVC的使用
KVC
的全称是Key-Value Coding
,也就是键值编码
,我们可以通过一个key来设置或获取某个属性的值。KVC
所用到的API如下:
// 通过key设置属性值
- (void)setValue:(id)value forKeyPath:(NSString *)keyPath;
- (void)setValue:(id)value forKey:(NSString *)key;
// 通过key获取属性值
- (id)valueForKeyPath:(NSString *)keyPath;
- (id)valueForKey:(NSString *)key;
我们看到设置和获取属性值的方法都有2个,一个是key
一个是keyPath
这两个有什么区别呢?直接看下面示例吧,我们先定义如下2个类:
// Dog类
#import <Foundation/Foundation.h>
@interface Dog : NSObject
@property (nonatomic , assign) NSInteger age;
@end
// Student类
#import <Foundation/Foundation.h>
#import "Dog.h"
@interface Student : NSObject
@property (nonatomic , strong) NSString *name;
@property (nonatomic , strong) Dog *dog;
@end
如果我们要设置或获取Student
实例对象的name
属性值,通过key
或keyPath
方式都是一样的:
- (void)test{
self.stu1 = [[Student alloc] init];
self.stu1.dog = [[Dog alloc] init];
[self.stu1 setValue:@"Jack" forKey:@"name"];
NSLog(@"通过key的方式--%@",[self.stu1 valueForKey:@"name"]);
[self.stu1 setValue:@"Bob" forKeyPath:@"name"];
NSLog(@"通过keyPath的方式--%@",[self.stu1 valueForKeyPath:@"name"]);
}
但是如果我们要设置或获取Student
实例对象的dog
的age
属性值,那就只能通过keyPath
的方式了。此时如果还是使用key
的方式设置属性值的话就会抛出setValue:forUndefinedKey:
的异常。
- (void)test{
self.stu1 = [[Student alloc] init];
self.stu1.dog = [[Dog alloc] init];
[self.stu1 setValue:@5 forKeyPath:@"dog.age"];
NSLog(@"%@",[self.stu1 valueForKeyPath:@"dog.age"]);
}
2 KVC设置属性值的流程
我们以setValue:forKey:
为例,KVC
设置属性值的整个流程如下图所示:
比如我们执行
[self.stu1 setValue:@"Jack" forKey:@"name"];
这句代码,其底层执行流程如下:
- 首先找到
self.stu1
的isa
指向的类对象,在类对象的方法列表中按照setName
、_setName
的顺序进行查找(也就是先查找看有没有setName
这个方法,没有的话再查找有没有_setName
方法)。 - 前面如果查找到了方法,就调用方法。
- 前面如果没有查找到方法,底层就会调用
Student
类的+ (BOOL)accessInstanceVariablesDirectly;
方法。如果返回值是YES
就表示允许直接访问类的成员变量,返回NO
表示不允许。这个方法是需要我们在Student
类中重写的,由开发者来决定是否允许访问成员变量,如果不重写,默认是返回YES。 - 如果上一步返回的是
NO
,那会直接抛出异常setValue:forUndefinedKey:
。 - 如果返回的是
YES
的话,那就会按照_name
、_isName
、name
、isName
这样一个顺序来查找类对象中的成员属性列表,如果找到了就直接赋值;如果没有找到就抛出异常setValue:forUndefinedKey:
。
3 KVC获取属性值的流程
KVC
获取属性值的流程图如下:
比如我们执行[self.stu1 valueForKey:@"name"];
这句代码,其底层执行流程如下:
- 首先找到self.stu1的isa指向的类对象,在类对象的方法列表中按照
getName
、name
、isName
、_name
的顺序进行查找。 - 上一步如果查找到了方法,就直接调用方法。
- 如果没有找到方法,就会调用
Student
类的+ (BOOL)accessInstanceVariablesDirectly;
方法,看其返回的是YES还是NO; - 如果上一步返回的是
NO
,就直接抛出异常valueForUndefinedKey:
。 - 如果返回的是
YES
,就会按照_name
、_isName
、name
、isName
这样一个顺序来查找类对象中的成员属性列表,如果找到了就直接取值;如果没有找到就抛出异常valueForUndefinedKey:
。