KVC详解

2018-08-24  本文已影响174人  蜗牛非牛

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:

  1. 首先按get<Key>、<key>、is<Key>的顺序查找getter方法,找到直接调用。如果是bool、int等内建值类型,会做NSNumber的转换。

  2. 上面的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:方法。

  1. 还没查到,那么查找countOf<Key>、enumeratorOf<Key>、memberOf<Key>:格式的方法。如果这三个方法都找到,那么就返回一个可以响应NSSet所有方法的代理集合(collection proxy object)。发送给这个代理集合(collection proxy object)的NSSet消息方法,就会以countOf<Key>、enumeratorOf<Key>、memberOf<Key>:组合的形式调用。

  2. 还是没查到,那么如果类方法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

上一篇 下一篇

猜你喜欢

热点阅读