疑问
平时项目中在用KVC的时候,大多只是简单的调用 setValue forKey
或者 valueForKey
做一些赋值与取值操作。但是具体的过程真的是我们想的那样么,所以我们通过一个demo一起详细的验证一下。
理论经验
-(void)setValue:(id)value forKey:(NSString)key
功能:使用一个字符串标示符给一个对象的属性赋值.它支持普通对象和集合对象
具体调用过程,这里是引用原作者的话并加以自己的理解与修改
KVC底层实现原理
KVC之setValue:forKey原理
1.首先去调用方法的对象的类中查找与key相匹配的set访问器方法,如果方法参数的类是一个对象指针类型,就会简单的执行这个方法,并传入对应的参数.如果方法的参数类型是NSNumber或NSValue的对应的基本类型(非空),先把它转换为基本数据类,再执行方法,传入转换后的数据.这里注意,若它的参数类型不是一个对象指针类型,但值为nil,就执行
setNilValueForKey:
方法,setNilValueForKey:
方法默认实现是产生一个NSInvalidArgumentException的异常,但你可重写这个方法.
2.如果没有对应的访问器set方法,如果调用者对象的类的+accessInstanceVariablesDirectly方法返回YES,那么就查找这个接受者的与key相匹配的实例变量(匹配顺序依次为_key,_isKey,key,isKey):比如:key为age,只要属性存在_age,_isAge,age,isAge中的其中一个就认为已经匹配上,如果找到这样的一个实例变量,并且的类型是一个对象指针类型,首先released对象上的旧值,然后把传入的新值retain后的传入的值赋值该成员变量,如果方法的参数类型是NSNumber或NSValue的对应的基本类型,先把它转换为基本数据类,再执行方法,传入转换后的数据.
3.如果访问器方法和实例变量都没有找到,执行
setValue:forUndefinedKey:
方法,该方法的默认实现是产生一个 NSUndefinedKeyException 类型的异常,但是可重写setValue:forUndefinedKey:方法来避免
一般在做简单的网络请求后的数据转模型时,经常重写此方法来避免一些不必要的崩溃
实践得真知
Student.h
@interface Student : NSObject
@property(nonatomic,copy)NSString *name;
@property(nonatomic,assign)NSInteger age;
@end
Student.m
@interface Student(){
NSString *_name;
NSInteger _age;
}
@end
@implementation Student
@synthesize name = _name;
@synthesize age = _age;
-(void)setName:(NSString *)name{
_name = name;
NSLog(@"name -- %@",_name);
}
- (void)setAge:(NSInteger)age{
_age = age;
NSLog(@"age -- %ld",(long)age);
}
- 验证上述第一点
Student * stu = [Student new];
[stu setValue:@"ZZZ" forKey:@"name"];
[stu setValue:@18 forKey:@"age"];
[stu setValue:nil forKey:@"name"];
输出结果:
Student * stu = [Student new];
[stu setValue:nil forKey:@"age"];
输出结果:
- 验证上述第二点,可以针对上述结论区分验证
@interface Student(){
// NSString *_teaName;
NSString *_isTeaName;
// NSString *teaName;
// NSString *isTeaName;
}
@end
[stu setValue:@"XXX" forKey:@"teaName"];
// NSLog(@"1-----%@",[stu valueForKey:@"_teaName"]);
NSLog(@"2-----%@",[stu valueForKey:@"_isTeaName"]);
// NSLog(@"3-----%@",[stu valueForKey:@"teaName"]);
// NSLog(@"4-----%@",[stu valueForKey:@"isTeaName"]);
输出结果:
- 如若1.2步都没有找到合适的key,在没有重写
- (void)setValue:(id)value forUndefinedKey:(NSString *)key
的情况下就直接报错了,开发中经常遇到,这里就略过了。
PS
本文重在验证理解确认,对于自己也算加深一下记忆吧,demo比较简单可自行尝试一下。如有错误或者遗漏请指出。
引用请阅读:传送门