iOS 关于kvc的一些记录

一:KVC是什么?

KVC是Key Value Coding的简称,即键值编码,提供一种机制来间接访问对象的属性。而不是通过调用Setter、Getter方法访问,KVC的方法定义在Foundation/NSKeyValueCoding中,是一个NSObject的扩展,任何继承NSObject的类都包含此方法。简单的来说,KVC让我们能够使用属性的字符串名称来设置、读取属性的值,而不是通过.语法实现。

二:KVC能够干什么?

  • 对私有变量进行赋值
  • 字典转模型

三:KVC的四个常用方法

//  根据属性名称key设置值
- (void)setValue:(nullable id)value forKey:(NSString *)key;
//  根据属性路径名称keyPath设置值
- (void)setValue:(nullable id)value forKeyPath:(NSString *)keyPath;
//  根据属性名称key获取值
- (nullable id)valueForKey:(NSString *)key;
//  根据属性路径名称keyPath获取值
- (nullable id)valueForKeyPath:(NSString *)keyPath;
1:KVC是如何获取值?之- (nullable id)valueForKey:(NSString *)key; 方法

其实这些方法在Foundation框架NSKeyValueCoding.h中都有对方法的解析
关于- (nullable id)valueForKey:(NSString *)key;获取属性值的调用过程如下:

1: 首先查找与其名称匹配的几个方法,判断调用顺序为:get<key>,<key>,is<key>
2:如果以上三个方法均没有,则调用+(BOOL)accessInstanceVariablesDirectly;(是否直接访问成员变量),返回YES则直接访问成员变量,返回NO,则调用- (id)valueForUndefinedKey:(NSString *)key指定一个值返回,不重写此方法默认崩溃
3:直接访问成员变量,也会顺序判断调用:_<key>、_is<key>、<key>、is<key>,如果没有找到这些成员变量,则调用- (id)valueForUndefinedKey:(NSString *)key指定一个值返回,不重写此方法默认崩溃

代码测试1:匹配方法
在TestKVCModel类中实现如下方法

@implementation TestKVCModel

//1:  获取kvc最高级
- (NSString *)getTestName {
    NSLog(@"getTestName");
    return @"xxx";
}
//2:  获取第二级
- (NSString *)testName {
    NSLog(@"testName");
    return @"xxx";
}
//3:  获取第三级
- (NSString *)isTestName {
    NSLog(@"isTestName");
    return @"xxx";
}
//  是否允许直接方法成员变量
+ (BOOL)accessInstanceVariablesDirectly {
    NSLog(@"accessInstanceVariablesDirectly");
    return NO;
}
//  没有访问到,默认返回值,不实现此方法,默认崩溃
- (id)valueForUndefinedKey:(NSString *)key {
    NSLog(@"valueForUndefinedKey %@", key);
    return @"AAAAA";
}

使用- (nullable id)valueForKey:(NSString *)key;方法访问testName的值

TestKVCModel *model = [[TestKVCModel alloc] init];
NSLog(@"%@", [model valueForKey:@"testName"]);
多次输出结果:
1:
2019-08-21 10:04:58.378737+0800 TestKVC[47554:2700945] getTestName
2019-08-21 10:04:58.378872+0800 TestKVC[47554:2700945] xxx
2:注释- (NSString *)getTestName方法
2019-08-21 10:06:02.375669+0800 TestKVC[47614:2702788] testName
2019-08-21 10:06:02.375849+0800 TestKVC[47614:2702788] xxx
3:注释- (NSString *)getTestName和- (NSString *)testName方法
2019-08-21 10:06:38.428103+0800 TestKVC[47654:2703877] isTestName
2019-08-21 10:06:38.428253+0800 TestKVC[47654:2703877] xxx
4:同时注释三个方法
2019-08-21 10:07:15.966436+0800 TestKVC[47693:2705005] accessInstanceVariablesDirectly
2019-08-21 10:07:15.966592+0800 TestKVC[47693:2705005] accessInstanceVariablesDirectly
2019-08-21 10:07:15.966689+0800 TestKVC[47693:2705005] valueForUndefinedKey testName
2019-08-21 10:07:15.966769+0800 TestKVC[47693:2705005] AAAAA

测试结果:我们分别注释TestKVCModel中的方法,可以观察其调用顺序,其调用顺序依次为- (NSString *)getTestName -> - (NSString *)testName -> - (NSString *)isTestName
注释上面三个方法默认返回值AAAAA,同时注释- (id)valueForUndefinedKey:(NSString *)key方法,程序会崩溃

接下来,我们为TestKVCModel类添加testName实例变量,+ (BOOL)accessInstanceVariablesDirectly返回YES允许直接访问实列变量
代码测试2:匹配实例变量
在TestKVCModel.h中添加三个属性

// 使用@property方式申明,会默认生成_key的实列变量,并默认实现getter和setter方法
@interface TestKVCModel : NSObject {
    NSString *_testName;
    NSString *testName;
    NSString *isTestName;
}
在TestKVCModel.m中,设置初始值,允许直接方法成员变量
- (instancetype)init
{
    self = [super init];
    if (self) {
        _testName = @"_testName";
        testName = @"testName";
        isTestName = @"isTestName";
    }
    return self;
}
//  是否允许直接方法成员变量
+ (BOOL)accessInstanceVariablesDirectly {
    NSLog(@"accessInstanceVariablesDirectly");
    return YES;
}
//  没有访问到,默认返回值,不实现此方法,默认崩溃
- (id)valueForUndefinedKey:(NSString *)key {
    NSLog(@"valueForUndefinedKey %@", key);
    return @"AAAAA";
}

使用- (nullable id)valueForKey:(NSString *)key;方法访问testName的值

 TestKVCModel *model = [[TestKVCModel alloc] init];
 NSLog(@"%@", [model valueForKey:@"testName"]);
多次输出结果:
1:
2019-08-21 10:17:54.136509+0800 TestKVC[48278:2719585] accessInstanceVariablesDirectly
2019-08-21 10:17:54.136658+0800 TestKVC[48278:2719585] _testName
2:注释_testName
2019-08-21 10:18:26.850766+0800 TestKVC[48323:2720715] accessInstanceVariablesDirectly
2019-08-21 10:18:26.850964+0800 TestKVC[48323:2720715] testName
3:注释_testName和testName
2019-08-21 10:18:56.617613+0800 TestKVC[48362:2721753] accessInstanceVariablesDirectly
2019-08-21 10:18:56.617764+0800 TestKVC[48362:2721753] isTestName
4:全部注释
2019-08-21 10:19:24.119137+0800 TestKVC[48401:2722593] accessInstanceVariablesDirectly
2019-08-21 10:19:24.119269+0800 TestKVC[48401:2722593] accessInstanceVariablesDirectly
2019-08-21 10:19:24.119399+0800 TestKVC[48401:2722593] valueForUndefinedKey testName
2019-08-21 10:19:24.119510+0800 TestKVC[48401:2722593] AAAAA

测试结果:我们分别注释TestKVCModel中的实列变量,可以观察其调用顺序,其调用顺序依次为_testName -> testName -> isTestName注释上面三个方法默认返回值AAAAA,同时注释-(id)valueForUndefinedKey:(NSString *)key方法,程序会崩溃

另外,- (nullable id)valueForKey:(NSString *)key也适用于类方法
代码测试3:匹配类方法
在TestKVCModel类中添加类方法如下

//1:  获取kvc最高级
+ (NSString *)getTestName {
    NSLog(@"getTestName");
    return @"xxx11";
}
//2:  获取第二级
+ (NSString *)testName {
    NSLog(@"testName");
    return @"xxx11";
}
//3:  获取第三级
+ (NSString *)isTestName {
    NSLog(@"isTestName");
    return @"xxx11";
}
//4:没有访问到,默认返回值,不实现此方法,默认崩溃
+ (id)valueForUndefinedKey:(NSString *)key {
    NSLog(@"valueForUndefinedKey %@", key);
    return @"AAAAA11";
}

使用类方法调用- (nullable id)valueForKey:(NSString *)key;访问testName的值

NSLog(@"%@", [TestKVCModel valueForKey:@"testName"]);
输出结果:
1:
2019-08-21 09:30:49.017583+0800 TestKVC[45743:2657198] getTestName
2019-08-21 09:30:49.017710+0800 TestKVC[45743:2657198] xxx11
2:注释+ (NSString *)getTestName方法
2019-08-21 10:00:02.245279+0800 TestKVC[47231:2691526] testName
2019-08-21 10:00:02.245431+0800 TestKVC[47231:2691526] xxx11
3:注释+ (NSString *)getTestName和+ (NSString *)testName方法
2019-08-21 10:00:43.002210+0800 TestKVC[47274:2692668] isTestName
2019-08-21 10:00:43.002343+0800 TestKVC[47274:2692668] xxx11
4:同时注释三个方法
2019-08-21 10:03:06.344779+0800 TestKVC[47426:2696736] valueForUndefinedKey testName
2019-08-21 10:03:06.344918+0800 TestKVC[47426:2696736] AAAAA11

测试结果:与代码测试1的实列方法调用一致,使用场景:在组件中获取pch文件或者其他组件定义的常量或者宏,可以在项目中为组件的类创建分类,使用- (nullable id)valueForKey:(NSString *)key;获取值

2:KVC是如何获取值?之- (nullable id)valueForKeyPath:(NSString *)keyPath; 方法

- (nullable id)valueForKeyPath:(NSString *)keyPath;方法是根据路径来获取值,可以获取对象中对象的实例变量值:
NSLog(@"%@", [model valueForKeyPath:@"subModel.nickName"]);
其查找的方式与- (nullable id)valueForKey:(NSString *)key;一致,先匹配方法,再匹配实列变量

代码测试1:获取对象中对象的值
新建一个SubTestKVCModel类,内容代码如下

SubTestKVCModel.m中
//1:  获取kvc最高级
- (NSString *)getNickName {
    NSLog(@"getNickName");
    return @"111";
}
//2:  获取第二级
- (NSString *)nickName {
    NSLog(@"nickName");
    return @"111";
}
//3:  获取第三级
- (NSString *)isNickName {
    NSLog(@"isNickName");
    return @"111";
}
//  是否允许直接方法成员变量
+ (BOOL)accessInstanceVariablesDirectly {
    NSLog(@"accessInstanceVariablesDirectly");
    return NO;
}
//  没有访问到,默认返回值,不实现此方法,默认崩溃
- (id)valueForUndefinedKey:(NSString *)key {
    NSLog(@"valueForUndefinedKey %@", key);
    return @"BBBBB";
}

TestKVCModel.h中
@property (nonatomic, strong) SubTestKVCModel *subModel;

使用- (nullable id)valueForKeyPath:(NSString *)keyPath;获取TestKVCModel中的subModel的nickName的值

TestKVCModel *model = [[TestKVCModel alloc] init];
model.subModel = [[SubTestKVCModel alloc] init];
NSLog(@"%@", [model valueForKeyPath:@"subModel.nickName"]);
输出结果:
2019-08-21 13:14:07.802015+0800 TestKVC[57293:2933510] getNickName
2019-08-21 13:14:07.802168+0800 TestKVC[57293:2933510] 111

也可以多层索引
NSLog(@"%@", [model valueForKeyPath:@"subModel.subSubModel.myName"]);

3:关于- (nullable id)valueForKeyPath:(NSString *)keyPath的其他使用

1:在数组中:取最小值、最大值、平均值、求和

    CGFloat sum = [[array valueForKeyPath:@"@sum.floatValue"] floatValue];
    CGFloat avg = [[array valueForKeyPath:@"@avg.floatValue"] floatValue];
    CGFloat max =[[array valueForKeyPath:@"@max.floatValue"] floatValue];
    CGFloat min =[[array valueForKeyPath:@"@min.floatValue"] floatValue];

2:在数组中:去除重复元素

NSArray *ary = @[@"a", @"b", @"c", @"d", @"a", @"e", @"c"];
    NSArray *resultAry = [ary valueForKeyPath:@"@distinctUnionOfObjects.self"];
    NSLog(@"%@", resultAry);
输出:
(
    c,
    d,
    e,
    a,
    b
)

更多实用方法见:https://www.jianshu.com/p/ff17a9619894

4:KVC是如何设置值的?之- (void)setValue:(nullable id)value forKey:(NSString *)key;方法

关于- (void)setValue:(nullable id)value forKey:(NSString *)key;设置属性值的调用过程如下:

1:查找与其名匹配的setKey方法
2:找到setKey方法,判断setKey的参数类型,如果类型非指针类型,且setValue设置的值为nil,则调用- (void)setNilValueForKey:(NSString *)key方法
3:如果没有找到setKey方法,则调用+ (BOOL)accessInstanceVariablesDirectly(是否允许直接方法成员变量),返回YES则直接访问成员变量,返回NO,则调用- (void)setValue:(id)value forUndefinedKey:(NSString *)ke,不重写此方法默认崩溃
4:直接访问成员变量,查找成员变量顺序:_<key>、_is<Key>、 <key>、 is<Key>

测试代码1

.h文件
@interface TestKVCModel : NSObject {
    int _testName;
    int _isTestName;
    int testName;
    int isTestName;
}
.m文件
//// kvc最高优先级
//- (void)setTestName:(NSString *)testName {
//    NSLog(@"setTestName %@", testName);
//}

// kvc最高优先级
- (void)setTestName:(int)testName {
    NSLog(@"setTestName int %d", testName);
}

// 实现setKey方法,判断setKey的参数如果非指针类型,且设置的值为nil,则调用此方法
- (void)setNilValueForKey:(NSString *)key {
    NSLog(@"setNilValueForKey %@", key);
}

// 没有访问到,调用此方法,没有重写此方法会崩溃
- (void)setValue:(id)value forUndefinedKey:(NSString *)key {
    NSLog(@"setValue forUndefinedKey %@", key);
}

//  没有实现setKey方法,是否允许直接方法成员变量
+ (BOOL)accessInstanceVariablesDirectly {
    NSLog(@"accessInstanceVariablesDirectly");
    return YES;
}

调用- (void)setValue:(nullable id)value forKey:(NSString *)key;方法设置值

TestKVCModel *model = [[TestKVCModel alloc] init];
 [model setValue:nil forKey:@"testName"];

依次注释代码,进行测试,测试结论如上

5:KVC是如何设置值的?之- (void)setValue:(nullable id)value forKeyPath:(NSString *)keyPath方法

- (void)setValue:(nullable id)value forKeyPath:(NSString *)keyPath;方法根据路径设置值,可以设置对象中对象的实例变量值:
[model setValue:nil forKeyPath:@"subModel.nickName"];
设置值的调用过程与- (void)setValue:(nullable id)value forKey:(NSString *)key;方法一致

测试代码1

TestKVCModel *model = [[TestKVCModel alloc] init];
model.subModel = [[SubTestKVCModel alloc] init];
[model setValue:nil forKeyPath:@"subModel.nickName"];
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 158,736评论 4 362
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,167评论 1 291
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,442评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,902评论 0 204
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,302评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,573评论 1 216
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,847评论 2 312
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,562评论 0 197
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,260评论 1 241
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,531评论 2 245
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,021评论 1 258
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,367评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,016评论 3 235
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,068评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,827评论 0 194
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,610评论 2 274
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,514评论 2 269

推荐阅读更多精彩内容

  • KVC是Key Value Coding的简称。它是一种可以通过字符串的名字(key)来访问类属性的机制。而不是通...
    153037c65b0c阅读 11,009评论 15 17
  • KVC是Key Value Coding的简称。它是一种可以通过字符串的名字(key)来访问类属性的机制。而不是通...
    _李恒阅读 670评论 0 0
  • 什么是KVC? KVC(Key-value coding)键值编码,单看这个名字可能不太好理解。其实是指iOS的开...
    萨缪阅读 4,547评论 1 13
  • 1. Basic methods KVC(Key-value coding)键值编码,就是指iOS的开发中,可以允...
    木小易Ying阅读 177评论 0 4
  • KVC(Key-value coding)键值编码,iOS的开发中,可以允许开发者通过Key名直接访问对象的属性,...
    CALayer_Sai阅读 2,386评论 0 4