版本记录
版本号 | 时间 |
---|---|
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
上面的顺序亲测,是按照所列顺序执行的。
后记
未完,待续~~
网友评论