美文网首页iOSiOS面试总结iOS进阶
iOS底层原理探索—KVC的本质

iOS底层原理探索—KVC的本质

作者: 劳模007_Mars | 来源:发表于2019-04-24 15:39 被阅读10次

    探索底层原理,积累从点滴做起。大家好,我是Mars。

    往期回顾

    iOS底层原理探索—OC对象的本质
    iOS底层原理探索—class的本质
    iOS底层原理探索—KVO的本质
    今天带领大家探索iOS之KVC的本质。

    KVC

    KVC全称是Key-Value Coding,键值编码,可以通过Key来访问和修改属性。
    KVC常用的API包括:

    //存值
    - (void)setValue:(id)value forKeyPath:(NSString *)keyPath;
    - (void)setValue:(id)value forKey:(NSString *)key;
    
    //取值
    - (id)valueForKeyPath:(NSString *)keyPath;
    - (id)valueForKey:(NSString *)key; 
    

    KVC的使用

    下面我们简单使用一下KVC:

    KVC简单使用.png
    通过代码可以看到,通过KVC我们给Person对象的age属性进行了赋值和取值的操作,同时还使用setValue: forKeyPath:方法做了一下相对复杂的Person对象中student的height属性做了赋值和取值。其中Person和Student的关系如下:
    KVC简单使用1.png

    setValue:forKey:只是根据属性名做赋值和取值的操作,而setValue: forKeyPath:方法则能根据key的路径做一些层级间的复杂访问操作。

    KVC底层原理

    KVC赋值

    KVC赋值的底层原理就是,当我们调用setValue: forKeyPath:方法后,系统会按照顺序查找两个方法:setKey_setKey,如果实现了这两个方法其中一个,那么就会传递参数,并且调用实现的方法。我们验证一下:
    我们通过代码来设置Person的age属性,并且在Person类里面依次分别实现setAge_setAge两个方法

    int main(int argc, const char * argv[]) {
        @autoreleasepool {
    
            Person *person = [[Person alloc] init];
            // 通过KVC修改age属性
            [person setValue:@10 forKey:@"age"];
    
            
        }
        return 0;
    }
    
    

    Person类.m文件中的实现:

    #import "Person.h"
    
    @implementation Person
    - (void)setAge:(int)age
    {
        NSLog(@"setAge: - %d", age);
    }
    
    //- (void)_setAge:(int)age
    //{
    //    NSLog(@"_setAge: - %d", age);
    //}
    @end
    

    第一次测试我们先注释掉_setAge方法,打印输出

    setAge:- 10
    

    第二次测试我们注释掉setAge方法,去掉_setAge的注释:

    #import "Person.h"
    
    @implementation Person
    //- (void)setAge:(int)age
    //{
    //    NSLog(@"setAge: - %d", age);
    //}
    
    - (void)_setAge:(int)age
    {
        NSLog(@"_setAge: - %d", age);
    }
    @end
    

    打印输出

    _setAge: - 10
    

    第三次测试我们将注释全部去掉:

    #import "Person.h"
    
    @implementation Person
    - (void)setAge:(int)age
    {
        NSLog(@"setAge: - %d", age);
    }
    
    - (void)_setAge:(int)age
    {
        NSLog(@"_setAge: - %d", age);
    }
    @end
    

    打印输出

    setAge:- 10
    

    通过测试我们可以验证我们上面的结论。
    如果没有实现setKey_setKey这两个方法的话,会继续调用accessInstanceVariablesDirectly方法,这个方法的返回值是BOOL类型,;如果返回值是NO,那么会调用setValue:forUndefinedKey:方法,并抛出异常NSUnknownKeyException的异常,赋值失败。如果返回值是YES,那么会继续按照顺序依次查找按照_key_isKeykeyisKey这四个成员变量,如果找到其中某个成员变量,完成赋值。如果四个成员变量全部未找到,那么会调用setValue:forUndefinedKey:方法,并抛出异常NSUnknownKeyException的异常,赋值失败。
    值得注意的是,accessInstanceVariablesDirectly方法返回值默认是YES。

    我们用流程图总结如下:


    KVC赋值流程.png

    KVC取值

    KVC取值流程和赋值流程一样,我们直接用流程图总结一下:


    KVC取值流程图.png

    关于KVC的底层原理探索我们告一段落,如有疑问,欢迎在评论区留言。

    更多技术知识请关注公众号
    iOS进阶


    iOS进阶.jpg

    相关文章

      网友评论

        本文标题:iOS底层原理探索—KVC的本质

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