美文网首页A知识点2
KVC解析(二) —— 不可不知的赋值深层次原理

KVC解析(二) —— 不可不知的赋值深层次原理

作者: 刀客传奇 | 来源:发表于2017-09-08 22:26 被阅读17次

    版本记录

    版本号 时间
    V1.0 2017.09.08

    前言

    KVC相信大家再熟悉不过了,键值编码,可以解决很多问题,包括视图上的给UITextField占位文字颜色大小进行设置等等,还有很多地方可以用KVC,接下来几篇我们就深度解析一下KVC。总结一下,就是指iOS的开发中,可以允许开发者通过Key名直接访问对象的属性,或者给对象的属性赋值。而不需要调用明确的存取方法。这样就可以在运行时动态地访问和修改对象的属性。而不是在编译时确定,这也是iOS开发中的黑魔法之一。还是老规矩,由面到点,由浅到深,希望对大家有所帮助。感兴趣的可以看我写的另外几篇文章。
    1. KVC解析(一) —— 基本了解

    KVC赋值

    赋值的过程看似很简单,就是找到对应的key,然后将一个已经准备好的值赋过去,但是你知道深层次的原理吗?方法- (nullable id)valueForKey:(NSString *)key;中的key系统内部和底层是按照什么逻辑找的呢?下面就让我们看一下。

    下面就是寻找key的内部流程。以[self setValue:@"小明" forKey:@"name"];这句代码作为例子进行说明。

    • 程序优先调用set<Key>:属性值方法,代码通过setter方法完成设置。注意,这里的<key>是指成员变量名,首字母大小写要符合KVC的命名规则,下同。

    • 如果没有找到setName:方法,KVC机制会检查+ (BOOL)accessInstanceVariablesDirectly方法有没有返回YES,默认该方法会返回YES,如果你重写了该方法让其返回NO的话,那么在这一步KVC会执行setValue:forUndefinedKey:方法,不过一般开发者不会这么做。所以KVC机制会搜索该类里面有没有名为_<key>的成员变量,无论该变量是在类接口处定义,还是在类实现处定义,也无论用了什么样的访问修饰符,只在存在以_<key>命名的变量,KVC都可以对该成员变量赋值。

    • 如果该类即没有set<key>:方法,也没有_<key>成员变量,KVC机制会搜索_is<Key>的成员变量。

    • 和上面一样,如果该类即没有set<Key>:方法,也没有_<key>_is<Key>成员变量,KVC机制再会继续搜索<key>is<Key>的成员变量。再给它们赋值。

    • 如果上面列出的方法或者成员变量都不存在,系统将会执行该对象的setValue:forUndefinedKey:方法,默认是抛出异常。

    • 特别需要注意的是:如果开发者想让这个类禁用KVC里,那么重写+ (BOOL)accessInstanceVariablesDirectly方法让其返回NO即可,这样的话如果KVC没有找到set<Key>:属性名时,会直接用setValue:forUndefinedKey:方法。

    下面结合例子进行说明

    1. + (BOOL)accessInstanceVariablesDirectly 返回NO实例

    #import "JJKVCSetValueVC.h"
    
    @interface JJKVCSetValueVC ()
    
    @end
    
    @implementation JJKVCSetValueVC
    
    {
        NSString *name;
    }
    
    #pragma mark - Override Base Function
    
    - (void)viewDidLoad
    {
        [super viewDidLoad];
        
        [self setValue:@"小明" forKey:@"name"];
        NSLog(@"name = %@", name);
    }
    
    + (BOOL)accessInstanceVariablesDirectly
    {
        return NO;
    }
    
    - (void)setValue:(id)value forUndefinedKey:(NSString *)key
    {
        NSLog(@"没有找到该key对应的属性会抛出异常");
    }
    
    @end
    

    下面看输出结果

    2017-09-08 21:50:31.966 JJOC[8051:216850] 没有找到该key对应的属性会抛出异常
    2017-09-08 21:50:31.966 JJOC[8051:216850] name = (null)
    

    这里+ (BOOL)accessInstanceVariablesDirectly返回了NO,也就是说找不到属性的setter方法,那么不会再去找实例变量,所以会输出上面的结果。所以name输出结果就是null

    2. name

    看示例代码

    #import "JJKVCSetValueDemoVC.h"
    
    @interface JJKVCSetValueDemoVC ()
    
    @end
    
    @implementation JJKVCSetValueDemoVC
    
    {
        NSString *name;
    }
    
    #pragma mark - Override Base Function
    
    - (void)viewDidLoad
    {
        [super viewDidLoad];
        
        [self setValue:@"小明" forKey:@"name"];
        NSLog(@"name = %@", name);
    }
    
    @end
    

    看输出结果

    2017-09-08 21:58:04.150 JJOC[8265:221362] name = 小明
    

    3. _name

    看示例代码

    #import "JJKVCSetValueDemoVC.h"
    
    @interface JJKVCSetValueDemoVC ()
    
    @end
    
    @implementation JJKVCSetValueDemoVC
    
    {
        NSString *_name;
    
    }
    
    #pragma mark - Override Base Function
    
    - (void)viewDidLoad
    {
        [super viewDidLoad];
        
        [self setValue:@"小明" forKey:@"name"];
        NSLog(@"name = %@", _name);
    }
    
    @end
    

    看输出结果

    2017-09-08 22:00:44.908 JJOC[8417:223934] name = 小明
    

    4. _isName

    看输出结果

    #import "JJKVCSetValueDemoVC.h"
    
    @interface JJKVCSetValueDemoVC ()
    
    @end
    
    @implementation JJKVCSetValueDemoVC
    
    {
        NSString *_isName;
    
    }
    
    #pragma mark - Override Base Function
    
    - (void)viewDidLoad
    {
        [super viewDidLoad];
        
        [self setValue:@"小明" forKey:@"name"];
        NSLog(@"name = %@", _isName);
    }
    
    @end
    

    看输出结果

    2017-09-08 22:03:21.038 JJOC[8581:226017] name = 小明
    

    5. isName

    看示例代码

    #import "JJKVCSetValueDemoVC.h"
    
    @interface JJKVCSetValueDemoVC ()
    
    @end
    
    @implementation JJKVCSetValueDemoVC
    
    {
        NSString *isName;
    
    }
    
    #pragma mark - Override Base Function
    
    - (void)viewDidLoad
    {
        [super viewDidLoad];
        
        [self setValue:@"小明" forKey:@"name"];
        NSLog(@"name = %@", isName);
    }
    
    @end
    

    看输出结果

    2017-09-08 22:13:24.258 JJOC[8797:231346] name = 小明
    

    下面就是我们这个属性和实例变量的调用顺序。

    • self.name
    • _name
    • _isName
    • name
    • isName

    上面的顺序亲测,是按照所列顺序执行的。

    后记

    未完,待续~~

    相关文章

      网友评论

        本文标题:KVC解析(二) —— 不可不知的赋值深层次原理

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