KVC进阶(一)

前言

KVC(Key Value Coding)技术可以不直接访问对象属性的gettersetter而获取属性值\给属性赋值

- (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)中的方法,看看他的运行过程吧

JKTest
ViewController

说明:以上两张图片中的代码都在ViewController.m中,因为屏幕不够长无法一次截取

先说一下为什么要在JKTest中写@synthesize。在同时实现settergetter时,类中的@property不会自动生成带下滑线的成员变量(分类中是不能生成,注意区分),所以这里要手动生成。当然,这里也可以直接在类中定义成员变量用于传值,看个人习惯。

接着我们看编译器输出结果,显然会自动调用属性对应的settergetter。但同时可以发现+ (BOOL)accessInstanceVariablesDirectly好像并没有调用,将代码稍作修改再次运行

ViewController

终于如愿以偿的看到+ (BOOL)accessInstanceVariablesDirectly,这个方法默认返回为YES。当返回YESkey值不是object时,会自动匹配_object(返回NO不会进行匹配)。但是问题又来了,这次object对应的settergetter并没有调用,真的是赋值给object而不是_object吗?毕竟_object本来就是存在的,为了验证这一点,再来做一次输出

ViewController

显然,object确实已被赋值,并且test.object仍然没有执行objectgetter,那么object究竟是怎么被赋值\取值的?

很容易想到,要再加如下测试代码

JKTest

问题又来了,+ (BOOL)accessInstanceVariablesDirectly现在不会调用,虽然仍然返回YES(有没有开始抓狂。。)
这是因为找到了_object对应的settergetter,既然找到了,自然不用匹配。那么为什么test.object会有值?因为写了@synthesize object = _object,此时已经将test.object_object关联起来,_object并不是一个独立的成员变量。如果不用@synthesize object = _object,改为定义成员变量_object仍然可行,因为Xcode会默认将test.object与自定义成员变量_object关联

如果将@synthesize object = _object改为@synthesize object = _obj并且实现_objgettersetter,KVC这样写

[test setValue:@"KVC" forKey:@"_obj"];
[test valueForKey:@"_obj"];

输出结果正确。因为test.object_obj已经关联,而且实现了对应的gettersetter,不会执行+ (BOOL)accessInstanceVariablesDirectly

同样,如果在此基础上将@synthesize object = _obj改为成员变量_obj,KVC还能正常运行吗?显然不能,因为Xcode不会将test.object与自定义成员变量_obj关联起来,这样KVC就无法匹配到正确的key值


这个问题总算解释清楚了,现在回到第二版本

ViewController

- (void)setValue:(id)value forKey:(NSString *)key+ (BOOL)accessInstanceVariablesDirectly的调用顺序究竟是怎样的?

JKTest

可见,实际顺序是先到- (void)setValue:(id)value forKey:(NSString *)key执行super然后跳到+ (BOOL)accessInstanceVariablesDirectly,最后再回到- (void)setValue:(id)value forKey:(NSString *)key

你可能感兴趣的:(KVC进阶(一))