KVC内部执行过程分析

关于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 *)change context:(void *)context{
    
    NSLog(@"监听到%@对象的%@属性改变---%@",object,keyPath,change);
}
@end

执行结果:

KVC[93661:9846235] 监听到对象的age属性改变---{
    kind = 1;
    new = 20;
    old = 18;
}

通过以上结果我们可以看出 通过KVC修改属性会触发KVO这是肯定的,同时我们也应该注意到通过KVC修改成员变量也会触发KVO

通过KVC修改属性会触发KVO我们可能还理解,可能内部帮我们调用了set方法。而通过KVC修改成员变量也会触发KVO是什么情况?通过上一篇文章KVO内部的实现原理 我们知道了KVO的本质,KVO的触发是因为系统内部调用了willChangeValueForKeydidChangeValueForKey方法才会触发KVO,而setValue:forKey或者setValue:ForKeyPath内部是不是也帮助我们做了这些事情呢?

下面我们通过代码来验证我们的猜测:


KVC内部执行过程分析_第1张图片
image

执行结果:

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] 监听到对象的age属性改变---{
    kind = 1;
    new = 20;
    old = 0;
}
2018-05-14 19:06:01.204675+0800 KVC[95006:10000295] didChangeValueForKey--end-age

以上执行结果也验证了我们的猜测:通过KVC修改成员变量的值,系统内部帮助我们调用了willChangeValueForKeydidChangeValueForKey这两个方法,而且是在didChangeValueForKey 内部调用了 observeValueForKeyPath:方法.

也就是说
[p1 setValue:@20 forKeyPath:@"age"];大概等价于下面这段代码:

[p1 willChangeValueForKey:@"age"];
 p1->age = 20;
[p1 didChangeValueForKey:@"age"];

KVC的赋值和取值过程是怎样的?原理是什么

下面我们通过一张图来看看setValue:forKey的执行过程:

KVC内部执行过程分析_第2张图片

1.当系统执行到[p1 setValue:@20 forKeyPath:@"age"];方法的时候,系统首先会去查找setAge:方法,如果该方法不存在则继续去寻找'_setAge:'方法,如果找到则传递参数调用方法.

2.没有找到 则去查看- (BOOL) accessInstanceVariablesDirectly方法的返回值,如果是NO,执行步骤3

3.系统自动调用setValue:forUndefinedKey:方法并且抛出NSUnknownKeyException的异常.

3.如果- (BOOL) accessInstanceVariablesDirectly方法的返回值是YES,当然其默认值就是YES,则系统会按照_age_isAgeageisAge顺序查找成员变量,如果找到了则直接赋值,否则执行步骤3;

同理,我们通过下面的图来看看valueForKey:的执行过程

KVC内部执行过程分析_第3张图片

由于图片的逻辑和上图类似,并且非常的清晰明了,这里就不在画蛇添足

完....

你可能感兴趣的:(KVC内部执行过程分析)