概述:
KVC简称KeyValueCoding,是一个基于NSKeyValueCoding非正式协议的机制,可以允许开发者通过Key名直接访问对象的属性,或者给对象的属性赋值。而不需要调用明确的存取方法。这样就可以在运行时动态访问和修改对象的属性。而不是在编译时确定,这也是iOS开发中的黑魔法之一。很多高级的iOS开发技巧都是基于KVC实现的。
注:NSObject是定义了KVC的,所以继承NSObject的对象都支持KVC,基本上所有的OC对象都支持KVC。
赋值过程:
MyObject *obj = [[MyObject alloc] init];
[obj setValue:@"哈哈哈" forKey:@"name"];
先查找相关的赋值方法,查找顺序是:
set<Key>
_set<Key>
setIs<Key>
如果没有找到相关赋值方法则调用+(BOOL)accessInstanceVariablesDirectly方法, 判断是否可以直接访问成员变量。(系统默认是返回YES的,但可以重写)
返回NO:调用- (void)setValue:(id)value forUndefinedKey:(NSString *)key方法并崩溃。错误异常是NSUnknownKeyException。
如果返回YES: 继续找相关变量,查找顺序是:
_<key>
_is<Key>
key
is<Key>
如果赋值方法和成员变量都不存在,则调用- (void)setValue:(id)value forUndefinedKey:(NSString *)key 并抛出异常NSUnknownKeyException。
取值过程:
MyObject *obj = [[MyObject alloc] init];
NSLog(@"%@",[obj valueForKey:@"name”]);
先查找相关赋值方法,查找顺序是:
get<Key>
key
如果没有找到相关方法,则调用+ (BOOL)accessInstanceVariablesDirectly 判断是否可以直接访问成员变量。
如果返回NO,则调用valueForUndefinedKey:, 并抛出系统异常:NSUnknownKeyException。
如果返回YES,则继续查找相关变量,查找顺序是
_<key>
_is<Key>
<key>
is<Key>
如果取值方法或者成员变量都不存在,则调用valueForUndefinedKey:方法,并抛出异常:NSUnknownKeyException。
注:如果想禁用KVC,重写+ (BOOL)accessInstanceVariablesDirectly 方法让其返回NO即可,这样 如果KVC找不到set<Key>属性名时,则会直接调用- (void)setValue:(id)value forUndefinedKey:(NSString *)key 并抛出异常NSUnknownKeyException。
常用场景:
1、属性赋值
Boy *jack = [[Boy alloc] init];
jack.book = [[Book alloc] init];
[jack.book setValue:@"iOS"forKeyPath:@"name"];//方式一
[jack setValue:@"C++"forKeyPath:@"book.name"];//方式二
2、添加私有成员变量
[jack setValue:@"182" forKeyPath:@"_height"];
3、KVC可以修改类的私有变量
可以看出利用KVC可以修改类的私有变量,可以修改IOS隐藏一些属性,如UITextField的placeHolderText默认style在需求中达不到要求,我们可以直接通过KVC快速定义自己的style,代码如下:
[textField setValue:[UIColor redColor] forKeyPath:@"placeholderLabel.textColor"];
4、KVC可以用来给model赋值
NSDictionary*dic = @{@"name":@"book",@"num":@"66"};
GoodModel *model = [[GoodModel alloc] init];
[model setValuesForKeysWithDictionary:dic];
NSLog(@"model.name : %@",model.name);
NSLog(@"model.num : %@",model.num);
注意:如果 model 属性和 dic 不匹配会怎样?
第一种情况,model多一个属性:这样程序没问题,model多出的属性会是nil
第二种情况,model少一个属性:程序会崩溃
第三种情况,model的属性名字和dic的key不匹配 : 程序会崩溃
解决方法: 第二种和第三种崩溃的解决办法是重写方法 -(void)setValue:(id)value forUndefinedKey:(NSString *)key
- (void)setValue:(id)value forUndefinedKey:(NSString*)key{
if([key isEqualToString:@"id"]){
self.goodId = (NSString*)value;
}
}