KVC内部执行过程分析

作者: Mark_Guan | 来源:发表于2018-05-14 19:30 被阅读26次

    关于KVC,我们先来看两道面试题:

    1.通过KVC修改属性会触发KVO么?
    2.KVC的赋值和取值过程是怎样的?原理是什么?
    
    

    KVC全称是Key Value Coding,俗称键值编码,它提供了一种间接访问其属性方法或成员变量的机制,也就是说可以通过字符串来访问对应的属性方法或成员变量。

    常见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的触发是因为系统内部调用了willChangeValueForKeydidChangeValueForKey方法才会触发KVO,而setValue:forKey或者setValue:ForKeyPath内部是不是也帮助我们做了这些事情呢?

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


    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] 监听到<Person: 0x604000018430>对象的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的执行过程:

    image

    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:的执行过程

    image

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

    完....

    相关文章

      网友评论

        本文标题:KVC内部执行过程分析

        本文链接:https://www.haomeiwen.com/subject/mxlmlftx.html