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:。