前言
KVC(Key Value Coding)技术可以不直接访问对象属性的getter
和setter
而获取属性值\给属性赋值
- (id)valueForKey:(NSString *)key;
- (void)setValue:(id)value forKey:(NSString *)key;
- (id)valueForKeyPath:(NSString *)keyPath;
- (void)setValue:(id)value forKeyPath:(NSString *)keyPath;
这几个方法相信大家都不会陌生,但KVC究竟是什么?
正文
在Foundation/NSKeyValueCoding
中可以看到,KVC是以非正式协议的方式出现的,里边的注释很详细这里不做搬运工了。来简单的实现一下NSObject(NSKeyValueCoding)
中的方法,看看他的运行过程吧
说明:以上两张图片中的代码都在ViewController.m中,因为屏幕不够长无法一次截取
先说一下为什么要在JKTest
中写@synthesize
。在同时实现setter
和getter
时,类中的@property
不会自动生成带下滑线的成员变量(分类中是不能生成,注意区分),所以这里要手动生成。当然,这里也可以直接在类中定义成员变量用于传值,看个人习惯。
接着我们看编译器输出结果,显然会自动调用属性对应的setter
和getter
。但同时可以发现+ (BOOL)accessInstanceVariablesDirectly
好像并没有调用,将代码稍作修改再次运行
终于如愿以偿的看到+ (BOOL)accessInstanceVariablesDirectly
,这个方法默认返回为YES
。当返回YES
且key值
不是object
时,会自动匹配_object
(返回NO
不会进行匹配)。但是问题又来了,这次object
对应的setter
和getter
并没有调用,真的是赋值给object
而不是_object
吗?毕竟_object
本来就是存在的,为了验证这一点,再来做一次输出
显然,object
确实已被赋值,并且test.object
仍然没有执行object
的getter
,那么object
究竟是怎么被赋值\取值的?
很容易想到,要再加如下测试代码
问题又来了,+ (BOOL)accessInstanceVariablesDirectly
现在不会调用,虽然仍然返回YES
(有没有开始抓狂。。)
这是因为找到了_object
对应的setter
和getter
,既然找到了,自然不用匹配。那么为什么test.object
会有值?因为写了@synthesize object = _object
,此时已经将test.object
与_object
关联起来,_object
并不是一个独立的成员变量。如果不用@synthesize object = _object
,改为定义成员变量_object
仍然可行,因为Xcode
会默认将test.object
与自定义成员变量_object
关联
如果将@synthesize object = _object
改为@synthesize object = _obj
并且实现_obj
的getter
和setter
,KVC这样写
[test setValue:@"KVC" forKey:@"_obj"];
[test valueForKey:@"_obj"];
输出结果正确。因为test.object
与_obj
已经关联,而且实现了对应的getter
和setter
,不会执行+ (BOOL)accessInstanceVariablesDirectly
。
同样,如果在此基础上将@synthesize object = _obj
改为成员变量_obj
,KVC还能正常运行吗?显然不能,因为Xcode
不会将test.object
与自定义成员变量_obj
关联起来,这样KVC就无法匹配到正确的key值
这个问题总算解释清楚了,现在回到第二版本
- (void)setValue:(id)value forKey:(NSString *)key
与+ (BOOL)accessInstanceVariablesDirectly
的调用顺序究竟是怎样的?
可见,实际顺序是先到- (void)setValue:(id)value forKey:(NSString *)key
执行super
然后跳到+ (BOOL)accessInstanceVariablesDirectly
,最后再回到- (void)setValue:(id)value forKey:(NSString *)key