KVC的实现原理

KVC是Key Value Coding的简称。它是一种可以通过字符串的名字(key)来访问类属性的机制。而不是通过调用Setter、Getter方法访问。KVC的方法定义在Foundation/NSKeyValueCoding中。

KVC使用的基本方法:

- (nullable id)valueForKey:(NSString*)key;//直接通过Key来取值

- (void)setValue:(nullable id)value forKey:(NSString*)key;//通过Key来设值

- (nullable id)valueForKeyPath:(NSString*)keyPath;//通过KeyPath来取值

- (void)setValue:(nullable id)value forKeyPath:(NSString*)keyPath;//通过KeyPath来设值

//默认返回YES,表示如果没有找到Set方法的话,会按照_key,_iskey,key,iskey的顺序搜索成员,设置成NO就不这样搜索

+ (BOOL)accessInstanceVariablesDirectly; 

//KVC提供属性值正确性验证的API,它可以用来检查set的值是否正确、为不正确的值做一个替换值或者拒绝设置新值并返回错误原因。 - (BOOL)validateValue:(inout id __nullable * __nonnull)ioValue forKey:(NSString *)inKey error:(out NSError **)outError;

 //如果Key不存在,且没有KVC无法搜索到任何和Key有关的字段或者属性,则会调用这个方法,默认是抛出异常。

 - (nullable id)valueForUndefinedKey:(NSString *)key;

- (void)setValue:(nullable id)value forUndefinedKey:(NSString *)key;//和上一个方法一样,但这个方法是设值。

- (void)setNilValueForKey:(NSString *)key;//如果你在SetValue方法时面给Value传nil,则会调用这个方法

- (NSDictionary *)dictionaryWithValuesForKeys:(NSArray *)keys;//输入一组key,返回该组key对应的Value,再转成字典返回,用于将Model转到字典。  

设值的实现步骤:

1.首先搜索是否有setKey:的方法(key是成员变量名,首字母大写),没有则会搜索是否有setIsKey:的方法。

2.如果没有找到setKey:的方法,此时看+ (BOOL)accessInstanceVariablesDirectly; (是否直接访问成员变量)方法。

若返回NO,则直接调用- (nullable id)valueForUndefinedKey:;(默认是抛出异常)。

若返回YES,按 _key、_iskey、key、isKey的顺序搜索成员名。

3.在第二步还没搜到的话就会调用- (nullable id)valueForUndefinedKey:方法。

验证一:如果实现setKey:方法则不会调用_setKey:和+ (BOOL)accessInstanceVariablesDirectly; 方法

创建一个类YYKVCModel,不声明属性,在.m文件同时实现以下三个方法:

+ (BOOL)accessInstanceVariablesDirectly{

    NSLog(@"accessInstanceVariablesDirectly");

    return YES;

}

- (void)_setTestName:(NSString *)testName{   

    NSLog(@"不被调用,因为实现了setTestName\n");

}

- (void)setTestName:(NSString *)testName{ 

     NSLog(@"setTestName调用\n");

}

用kvc赋值

YYKVCModel *model = [[YYKVCModel alloc] init];

 [model setValue:@"1223" forKey:@"testName"];

运行结果:只调用了setTestName方法。说明setTestName方法优先级高于_setTestName方法。

验证二:注释掉setTestName:方法,新增方法setIsTestName:

+ (BOOL)accessInstanceVariablesDirectly{

  NSLog(@"accessInstanceVariablesDirectly");

    return YES;

}

- (void)_setTestName:(NSString *)testName{ 

    NSLog(@"_setTestName被调用\n");

}

- (void)setIsTestName:(NSString *)testName{  

     NSLog(@"setIsTestName不被调用\n");

}

运行结果:调用了_setTestName方法,不调用setIsTestName方法。说明_setTestName的优先级高于setIsTestName。

验证三:注释掉_setTestName:方法添加成员变量

@interface YYKVCModel : NSObject {

    NSString *_testName;

}

运行结果:调用了setIsTestName:方法,_testName的值为null。说明setIsTestName方法优先级高于直接给_testName赋值。

验证四:注释掉setIsTestName:方法,为YYKVCModel增加以下成员变量

@interface YYKVCModel : NSObject {

    NSString *_testName;

    NSString *_isTestName;

}

运行结果:+ (BOOL)accessInstanceVariablesDirectly方法被调用,此时返回的是YES。_testName值变为1223,_isTestName值为null。说明直接给_testName赋值优先级高于直接给_isTestName赋值。

验证五:为YYKVCModel增加以下成员变量

@interface YYKVCModel : NSObject {

    NSString *_isTestName;

    NSString *testName;

}

运行结果:+ (BOOL)accessInstanceVariablesDirectly方法被调用,此时返回的是YES。_isTestName值变为1223,testName值为null。说明直接给_isTestName赋值优先级高于直接给testName赋值。

验证六:为YYKVCModel增加以下成员变量

@interface YYKVCModel : NSObject {

    NSString *testName;

    NSString *isTestName;

}

运行结果:+ (BOOL)accessInstanceVariablesDirectly方法被调用,此时返回的是YES。testName值变为1223,isTestName值为null。说明直接给testName赋值优先级高于直接给isTestName赋值。

验证七:注释 NSString *testName;

运行结果:+ (BOOL)accessInstanceVariablesDirectly方法被调用,此时返回的是YES。isTestName值变为1223。

验证八:在+ (BOOL)accessInstanceVariablesDirectly方法返回NO

运行结果:调用两次+ (BOOL)accessInstanceVariablesDirectly方法,然后调用- (void)setValue:(id)value forUndefinedKey:(NSString *)key方法,不实现方法将默认抛出异常,isTestName值为null。

若返回YES则只会调用一次。

由以上实验得出结论:

以上的代码中并没有声明testName这个属性,若声明属性会默认生成setter和getter方法,无法进行测试。

KVC的赋值本质上只是调用了属性的setter方法,setter方法会按照setKey、_setKey、setIsKey的优先级进行调用,还没有,则按_key、_isKey、key、isKey查找成员变量。

如果accessInstanceVariablesDirectly返回NO,则不会查找_key、_isKey、key、isKey,会直接调用- (void)setValue:(id)value forUndefinedKey:(NSString *)key。

若查找到isKey还是没找到,也会调用(void)setValue:(id)value forUndefinedKey:(NSString *)key,该方法默认会抛出异常。

KVC取值的实现:

1.按先后顺序搜索getKey、key、isKey、_getKey、_key五个方法,若某一个方法被实现,取到的即是方法返回的值,后面的方法不再运行。如果是BOOL或者Int等值类型, 会将其包装成一个NSNumber对象。

2.若这五个方法都没有找到,则会调用+ (BOOL)accessInstanceVariablesDirectly方法判断是否允许取成员变量的值。

若返回NO,直接调用- (nullable id)valueForUndefinedKey:(NSString *)key方法,默认是奔溃。

若返回YES,会按先后顺序取_key、_isKey、 key、isKey的值。

3.返回YES时,_key、_isKey、 key、isKey的值都没取到,调用- (nullable id)valueForUndefinedKey:(NSString *)key方法。

验证方法与上面设置值类似。

验证后得出结论:

在实验中同样没有用属性声明testName。在取值过程中按顺序看getKey、key、isKey、_getKey、_key五个方法是否实现,若实现了则取到的值为方法返回的值,所以本质上是按先后顺序调用了这五个getter方法,如果没有,则会询问+ (BOOL)accessInstanceVariablesDirectly方法能否直接取成员变量,若返回YES,则会按顺序取_key、_isKey、 key、isKey的值。

如果在+ (BOOL)accessInstanceVariablesDirectly中返回NO,或者取到isKey仍然取不到值,则会调用- (nullable id)valueForUndefinedKey方法,该方法中返回的值即为取到的值。

KVC取值的补充:

在取值的第一步结束第二步开始之前,还会判断获取的值是否是数组或者集合。

1.如果是数组(NSArray *)的话,当实现countOf方法和objectInAtIndex或AtIndexes中任意一个方法时则会返回一个数组.

具体写法:key为testArray

-(NSUInteger)countOfTestArray{

    return 3;

}

- (id)testArrayAtIndexes:(NSIndexSet *)indexs{

    return @[@(2), @(4), @(6)];

}

-(id)objectInTestArrayAtIndex:(NSUInteger)index{    

    return @(index);

}

KVC调用

YYKVCModel *model = [[YYKVCModel alloc] init];

id array = [model valueForKey:@"testArray"]; array类型为NSKeyValueArray

2.不可变数组还可以实现get<Key>:range:

如果是可变数组还可以实现:

-insertObject:in<Key>AtIndex:或者-insert<Key>:atIndexes:

-removeObjectFrom<Key>AtIndex:或者-remove<Key>AtIndexes:

-replaceObjectIn<Key>AtIndex:withObject:或者-replace<Key>AtIndexes:with<Key>:

3.集合需要实现以下三个方法:

-countOf<Key>

-enumeratorOf<Key>

-memberOf:<key>

属性是集合和数组的实现

KVC

KVC中的异常

1.获取值时找不到key

- (nullable id)valueForUndefinedKey:(NSString *)key;

2.设值时找不到key

- (void)setValue:(nullable id)value forUndefinedKey:(NSString *)key;

3.给不能设置nil的属性设置了nil。

定义一个属性

@property (nonatomic, assign) NSInteger num;

用kvc赋值

YYKVCModel *model = [[YYKVCModel alloc] init];

[model setValue:nil forKey:@"num"];

此时运行的话程序会崩溃,报错如下

重写- (void)setNilValueForKey:(NSString *)key方法后会发现不再奔溃,可知在该方法中默认抛出了异常。我们可以重写该方法做处理。

KVC处理非对象和自定义对象

KVC中返回的是一个id类型的对象,所以调用valueForKey:时如果是基本数据类型或者结构体,KVC会自动转成NSNumber类型或者NSValue类型,但是调用SetValue: forKey:时需要手动把基本数据类型或者结构体转成对象。

自定义对象需要我们自己保证类型的正确性。

KVC的属性验证

- (BOOL)validateValue:(inout id _Nullable __autoreleasing *)ioValue forKey:(NSString *)inKey error:(out NSError * _Nullable __autoreleasing *)outError;对应使用key的方式。

- (BOOL)validateValue:(inout id _Nullable __autoreleasing *)ioValue forKeyPath:(NSString *)inKeyPath error:(out NSError * _Nullable __autoreleasing *)outError;对应使用keyPath的方式。

KVC并不会自动调用该方法,需要我们手动调用

比如在设置值的时候我们可以判断键值的正确性,如果正确则继续操作,失败则不操作。验证失败后可以把错误存放在outError返回,这边的ioValue传入的也是指针,所以可以在需要时在验证方法中更改value,然后在外面设置。

KVC与容器类

 在使用KVO观察属性改变时,会发现如果观察可变数组时,对于添加或者移除元素时并不能接收到变化。因为KVO的本质是在setter方法上,添加上- (void)willChangeValueForKey:(NSString *)key和- (void)didChangeValueForKey:(NSString *)key方法来发送通知。此时用mutableArrayValueForKey:方法获取数组,在做增加删除操作就能接收到监听。

写法如:

[[self mutableArrayValueForKey:@"mutableArray"] addObject:@"1"];

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

推荐阅读更多精彩内容

  • KVC(Key-value coding)键值编码,单看这个名字可能不太好理解。其实翻译一下就很简单了,就是指iO...
    朽木自雕也阅读 1,554评论 6 1
  • KVC KVC定义 KVC(Key-value coding)键值编码,就是指iOS的开发中,可以允许开发者通过K...
    暮年古稀ZC阅读 2,140评论 2 9
  • 这两天晚上在学习腾讯公开课的视频,其中在1月18号的内容是KVC的底层实现。看完后简单记一下学习笔记。 1、[ob...
    ZzS_2d89阅读 544评论 0 1
  • 一、KVO的实现原理 KVO的全称是Key-ValueObserving(键值监听),可以用于监听某个对象属性值的...
    王的for阅读 1,048评论 1 0
  • KVC(Key-value coding)键值编码,单看这个名字可能不太好理解。其实翻译一下就很简单了,就是指iO...
    我的梦工厂阅读 891评论 1 8