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

版本记录

版本号 时间
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

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

后记

未完,待续~~

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • iOS KVC(一)基本了解iOS KVC (二) 不可不知的赋值深层次原理iOS KVC (三)不可不知的取值...
    奔跑吧小蚂蚁阅读 1,322评论 2 6
  • 1. Basic methods KVC(Key-value coding)键值编码,就是指iOS的开发中,可以允...
    木小易Ying阅读 201评论 0 4
  • 前言:往往会某项工具WORK,就想究其原理。本文先简单介绍KVC 一、KVC 简介 1.1 KVC 概述 1.KV...
    梦蕊dream阅读 941评论 0 2
  • KVC(Key-value coding)键值编码,iOS的开发中,可以允许开发者通过Key名直接访问对象的属性,...
    CALayer_Sai阅读 2,567评论 0 4
  • 上课已经是第二周了,每次都像个慢半拍的人一般,想想真是有点想笑。以前一直想写些当时的心境或是感悟之类的东西...
    Galsang敏阅读 667评论 2 1