关于KVC,我们先来看两道面试题:
1. 通过KVC修改属性会触发KVO么?
2. KVC技术是否违背了面向对象的这一思想?
3. KVC的赋值和取值过程是怎样的?原理是什么?
KVC全称是Key Value Coding
,俗称键值编码
,它提供了一种间接
访问其属性方法或成员变量的机制,也就是说可以通过字符串来访问对应的属性方法或成员变量。所以只要我们知道某个对象中的属性,就可以访问到,从这方面来说 KVC是有违背面向对象这一思想的。
常见API
动态赋值
setValue:forKey
只能给当前对象的属性赋值
setValue: forKeyPath:
还能给当前对象的属性的属性赋值
setValuesForKeysWithDictionary
传递一个字典给自己的所有属性赋值,其中属性名字必须和字典中key一致
setValue: forUndefinedKey:
当找不到某个Key的时候,系统就会自动调用该方法
动态取值
valueForKey
获取当前对象的属性的值
valueForKeyPath
也能获取当前对象的属性的属性的值
通过KVC修改属性会触发KVO么?
我们通过一段代码来看看第一道面试题目:
@interface Person : NSObject{
@public
int age;
}
@end
@implementation Person
@end
@interface ViewController ()
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
Person *p1 = [[Person alloc] init];
[p1 addObserver:self forKeyPath:@"age" options:NSKeyValueObservingOptionOld|NSKeyValueObservingOptionNew context:nil];
//通过KVC来改变age的值
[p1 setValue:@20 forKeyPath:@"age"];
}
-(void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context{
NSLog(@"监听到%@对象的%@属性改变---%@",object,keyPath,change);
}
@end
执行结果:
KVC[93661:9846235] 监听到<Person: 0x6040000065f0>对象的age属性改变---{
kind = 1;
new = 20;
old = 18;
}
通过以上结果我们可以看出 通过KVC修改属性会触发KVO这是肯定的,同时我们也应该注意到通过KVC修改成员变量也会触发KVO
通过KVC修改属性会触发KVO我们可能还理解,可能内部帮我们调用了set
方法。而通过KVC修改成员变量也会触发KVO是什么情况?通过上一篇文章KVO内部的实现原理 我们知道了KVO的本质,KVO的触发是因为系统内部调用了willChangeValueForKey
和didChangeValueForKey
方法才会触发KVO,而setValue:forKey
或者setValue:ForKeyPath
内部是不是也帮助我们做了这些事情呢?
下面我们通过代码来验证我们的猜测:
执行结果:
2018-05-14 19:05:57.508687+0800 KVC[95006:10000295] willChangeValueForKey---age
2018-05-14 19:05:57.508888+0800 KVC[95006:10000295] didChangeValueForKey--began-age
2018-05-14 19:06:01.204466+0800 KVC[95006:10000295] 监听到<Person: 0x604000018430>对象的age属性改变---{
kind = 1;
new = 20;
old = 0;
}
2018-05-14 19:06:01.204675+0800 KVC[95006:10000295] didChangeValueForKey--end-age
以上执行结果也验证了我们的猜测:通过KVC修改成员变量的值,系统内部帮助我们调用了willChangeValueForKey
和didChangeValueForKey
这两个方法,而且是在didChangeValueForKey
内部调用了 observeValueForKeyPath:
方法.
也就是说
[p1 setValue:@20 forKeyPath:@"age"];
大概等价于下面这段代码:
[p1 willChangeValueForKey:@"age"];
p1->age = 20;
[p1 didChangeValueForKey:@"age"];
KVC的赋值和取值过程是怎样的?原理是什么
下面我们通过一张图来看看setValue:forKey
的执行过程:
1.当系统执行到[p1 setValue:@20 forKeyPath:@"age"];
方法的时候,系统首先会去查找setAge:
方法,如果该方法不存在则继续去寻找'_setAge:'方法,如果找到则传递参数调用方法.
2.没有找到 则去查看- (BOOL) accessInstanceVariablesDirectly
方法的返回值,如果是NO,执行步骤3
3.系统自动调用setValue:forUndefinedKey:
方法并且抛出NSUnknownKeyException
的异常.
3.如果- (BOOL) accessInstanceVariablesDirectly
方法的返回值是YES,当然其默认值就是YES,则系统会按照_age
、_isAge
、age
、isAge
顺序查找成员变量,如果找到了则直接赋值,否则执行步骤3;
同理,我们通过下面的图来看看valueForKey:
的执行过程
由于图片的逻辑和上图类似,并且非常的清晰明了,这里就不在画蛇添足
完....