KVC(Key-valueCoding)
KVC键值编码,是对NSObjcet的扩展,分类名为:NSKeyValueCoding。是可以通过对象属性名称(Key)直接给属性值(value)进行赋值(coding),是系统提供的一套间接访问对象属性的机制,而不是通过调用set,与get方法访问。
KVC源码探讨
man setValue: forKey:
Person.h
#import <Foundation/Foundation.h>
@interface Person : NSObject{
NSString *height;
}
/** */
@property (nonatomic, copy) NSString *name;
@end
Man类
#import "Person.h"
@interface Man : Person
/** */
@property (nonatomic, copy) NSString *sex;
@end
#import "Man.h"
@implementation Man
+(BOOL)accessInstanceVariablesDirectly{
NSLog(@"accessInstanceVariablesDirectly");
return YES;
}
-(id)valueForUndefinedKey:(NSString *)key{
NSLog(@"出现异常,该key不存在%@",key);
return nil;
}
-(void)setValue:(id)value forUndefinedKey:(NSString *)key{
NSLog(@"出现异常,该key不存在%@",key);
}
@end
ViewController.m
- (void)viewDidLoad {
[super viewDidLoad];
self.view.backgroundColor = [UIColor whiteColor];
Man *man = [[Man alloc]init];
[man setValue:@"xiaolizi" forKey:@"name"];
NSString* height = [man valueForKey:@"name"];
NSLog(@"%@",height);
}
输出结果:
2018-08-24 17:24:40.593251+0800 KVC原理解析-18-8-24-3[8170:341077] xiaolizi
总结:程序优先调用setName方法,当找到之后,代码直接通过setter方法完成设置。
通过KVC对Man的父类people进行成员变量height赋值:
- (void)viewDidLoad {
[super viewDidLoad];
self.view.backgroundColor = [UIColor whiteColor];
Man *man = [[Man alloc]init];
[man setValue:@"180" forKey:@"height"];
NSString* height = [man valueForKey:@"height"];
NSLog(@"%@",height);
}
输出结果:
2018-08-24 17:06:48.842664+0800 KVC原理解析-18-8-24-3[7909:328924] accessInstanceVariablesDirectly
2018-08-24 17:06:48.842829+0800 KVC原理解析-18-8-24-3[7909:328924] accessInstanceVariablesDirectly
2018-08-24 17:06:48.842919+0800 KVC原理解析-18-8-24-3[7909:328924] 180
通过输出我们可以发现如果没有setHeight方法,KVC机制会检查+(BOOL)accessInstanceVariablesDirectly方法有没有返回YES,系统默认为YES,如果为YES则会继续沿着isa指针指向的类进行查找,顺序匹配变量名与_ <key>,_is<Key>,<key>,is<Key>,匹配到则设定其值,如果返回NO,结束查找。并调用setValue:forUndefinedKey:报异常。如果+(BOOL)accessInstanceVariablesDirectly方法返回值设置为NO,则直接调用setValue:forUndefinedKey:报异常。
man valueForKey:
首先按get<Key>、<key>、is<Key>的顺序查找getter方法,找到直接调用。如果是bool、int等内建值类型,会做NSNumber的转换。
上面的getter没有找到,查找countOf<Key>、objectIn<Key>AtIndex:、<Key>AtIndexes格式的方法。
如果countOf<Key>和另外两个方法中的一个找到,那么就会返回一个可以响应NSArray所有方法的代理集合(collection proxy object)。发送给这个代理集合(collection proxy object)的NSArray消息方法,就会以countOf<Key>、objectIn<Key>AtIndex:、<Key>AtIndexes这几个方法组合的形式调用。还有一个可选的get<Key>:range:方法。
还没查到,那么查找countOf<Key>、enumeratorOf<Key>、memberOf<Key>:格式的方法。如果这三个方法都找到,那么就返回一个可以响应NSSet所有方法的代理集合(collection proxy object)。发送给这个代理集合(collection proxy object)的NSSet消息方法,就会以countOf<Key>、enumeratorOf<Key>、memberOf<Key>:组合的形式调用。
还是没查到,那么如果类方法accessInstanceVariablesDirectly返回YES,那么按_<key>,_is<Key>,<key>,is<key>的顺序直接搜索成员名。
参考链接:https://blog.csdn.net/cewei711/article/details/80294048
参考链接:https://blog.csdn.net/qq_18505715/article/details/80205796