key-value coding

  • About key-value coding(关于KVC)

关键值编码是一种机制,使非正式协议NSKeyValueCoding对象采取提供间接访问性能,当一个对象遵守键值编码时,它的属性可以通过字符串参数通过简明的方式进行寻址,这种间接访问机制也可以直接访问实例变量

你通常使用访问器方法访问一个对象的属性。一个get访问器(或getter方法)返回一个属性的值。set访问器(或setter方法)设置一个属性的值。在Objective-C中,你也可以直接访问属性的实例变量。以任何这些方式访问对象属性都很简单,但需要调用特定于属性的方法或变量名,随着属性列表的增长或变化,访问这些属性的代码也必须随之改变。与此相反,键值兼容的对象提供了一个简单的消息传递接口,它的所有属性都是一致的

key-value coding 也是其它一些技术的基本概念,比如:kvo,core data......

  • Using Key-Value Coding Compliant Objects(使用键值对兼容对象)

一个对象如果是直接或者间接继承于NSObject,
并且都遵守NSKeyValueCoding 协议和一些基本方法的实现(这句话的意思如下,其实nsobject已经遵守了NSKeyValueCoding协议)


0410AF7F-1341-400A-ADD9-9143B03419E0.png

所以其实简单的来说只要满足第一个条件它就有kvc的功能

  • Key-Value Coding Fundamentals(KVC的基本概念)

  • Accessing Object Properties(访问对象的属性)

对象通常在接口声明中指定属性,而这些属性属于几个类别中的一个

  • 1 Attributes

这些都是一些简单的值,比如基本类型,NSString ,NSNumber,以及其它一些不可改变类型比如NSColor都是属于Attributes

  • 2 To-one relationships

这些是具有自身属性的可变对象, 对象的属性可以在不改变对象本身的情况下更改,其实可以理解这个属性为自定义对象(我自己的推测理解)

  • 3 To-many relationships

这些是集合对象。你通常使用NSArray或者NSSet持有这样一个集合,虽然自定义集合类也是可能的

把官网的示意图截取下来

1417FE9F-022F-4E3B-A2C5-F22C9233F679.png

为了保持封装性,一个对象通常提供的接口的属性访问器方法。对象的作者可以显式地写这些方法,也可以依赖编译器synthesize 自动地合成它们(set和get方法),无论哪种方法,在编译之前都得设置好属性名或者方法名

这是直接的,但缺乏灵活性。另一方面,键值兼容的对象提供了一种更一般的机制来使用字符串标识符访问对象的属性。
为了保持封装性,

  • Identifying an Object’s Properties with Keys and Key Paths

键是标识特定属性的字符串

[myAccount setValue:@(100.0) forKey:@"currentBalance"];
  • Getting Attribute Value Using key
  • 1. - (nullable id)valueForKey:(NSString *)key;
  • 2.- (nullable id)valueForKeyPath:(NSString *)keyPath;
  • 3.- (NSDictionary<NSString *, id> *)dictionaryWithValuesForKeys:(NSArray<NSString *> *)keys;
  • 4.- (nullable id)valueForUndefinedKey:(NSString *)key;

注意: 集合对象,比如NSArray,NSSet,NSDictionary,都不能包含为nil的值,所以如果你在设置值的时候,可以用NSNull对象,NSNull提供单实例表示对象属性的值为nil,默认的实现方法dictionaryWithValuesForKeys: 和setValuesForKeysWithDictionary:将自动转换 NSNull (字典里的参数) and nil (储存的属性)

当你使用一个键路径是处理属性时,如果键路径中的最后一个键是一个多对关系(也就是它引用集合),那么返回的值就是这个键对应所有值的一个集合,比如请求一个键路径的值transactions.payee那么返回一个数组,这个数组包含transactions下payee所有的值

  • Setting Attribute Values Using Keys
  • 1.- (void)setValue:(nullable id)value forKey:(NSString *)key;
  • 2.- (void)setValue:(nullable id)value forKeyPath:(NSString *)keyPath;
  • 3.- (void)setValue:(nullable id)value forUndefinedKey:(NSString *)key;
  • 4.- (void)setValuesForKeysWithDictionary:(NSDictionary<NSString *, id> *)keyedValues;

在默认的实现中,如果你要给非对象(比如int) 的属性设置nil,那么就会抛出一个异常,如果重写了setNilValueForKey,就不会奔溃,也可以在这个方法里改变值

  • Using Collection Operators(集合操作)

当你使用valueForKeyPath时,可以嵌入集合运算符在里面,集合运算符是一个小的关键字列表,前面是(@)

A6B7444C-84C7-4523-B054-08888C4846B6.png

集合操作展示三种基本行为类型

  • Aggregation Operators

以某种方式合并集合的对象,然后返回一个对象,该对象通常与右键路径中命名的属性的数据类型相匹配,但是@count是一个例外,它没有正确的关键路径和总是返回一个NSNumber实例,因为它返回的是集合的个数
1.@avg

NSMutableArray * arr=[[NSMutableArray alloc ]init];
Person * personOne = [[Person alloc ] init];
personOne.age=@"13";
[arr addObject:personOne];
Person * personTwo = [[Person alloc ] init];
personTwo.age=@"14";
[arr addObject: personTwo];
Person * personThree= [[Person alloc ] init];
personThree.age=@"15";
[arr addObject: personThree];
NSLog(@"平均值=%@",arr valueForKeyPath:@"@avg.age");
//14

2.@count
返回一个NSNumber实例集合中对象的数量

NSLog(@"集合个数=%@",arr valueForKeyPath:@"@count"); 
//3

3.@max
在由右键路径命名的集合条目中搜索并返回最大的条目。搜索进行比较使用比较的方法,如:许多基础类定义的,比如NSNumber类。因此,由右键路径指示的属性必须持有对该消息有意义响应的对象。搜索忽略空值集合条目

NSLog(@"最大值=%@",arr valueForKeyPath:@"@max.age");
//15

4.@min

NSLog(@"最小值=%@",arr valueForKeyPath:@"@min.age");
//13

5.@sum

NSLog(@"求和=%@",arr valueForKeyPath:@"@sum.age");
//42
  • Array Operators

返回与右键路径指示的对象的特定集合相对应的对象数组
1.@distinctUnionOfObjects
创建并返回包含与右键路径指定的属性相对应的集合的不同对象的数组

NSLog(@"返回数组下对象的某个属性的全部值,是个数组=%@",[arr valueForKeyPath:@"@distinctUnionOfObjects.age"]);
    /*
     =(
     15,
     13,
     14
     )

     */

注意:@distinctUnionOfObjects操作者会删除重复的对象,比如如果有三个14,那么就只返回一个14
2.@unionOfObjects
跟@distinctUnionOfObjects相反,它不会删除重复对象,如果有三个14,还是三个14全部返回

NSLog(@"返回数组下对象的某个属性的全部值,是个数组=%@",[arr valueForKeyPath:@"@unionOfObjects.age"]);
    /*
     =(
     13,
     14,
     14
     )

     
     */
  • Nesting Operators
  • Representing Non-Object Values(表示非对象值)

简单的总结几句话:setValue的时候如果不是value不是对象,那么转成对象,valueForKey的时候,因为返回的是id,如果是标量类型比如int,float都那么返回的是NSNumber ,把NSNumber转成int或者float,如果是结构体类型,那么返回的就是NSValue,把NSValue转成NSRect,NSPoint,NSSize,NSRange就行

  • Validating Properties(验证属性)

KVC提供了属性值,用来验证key对应的Value是否可用的方法
validateValue:forKey:error: 和validateValue:forKeyPath:error方法
这个方法的默认实现是去探索类里面是否有一个这样的方法:-(BOOL)validate<Key>:error:如果有这个方法,就调用这个方法来返回,没有的话就直接返回YES


@implementation Person


-(BOOL)validateAge:(id *)value error:(out NSError * _Nullable __autoreleasing *)outError{  //在implementation里面加这个方法,它会验证是否设了非法的value
    NSString* inValue = *value;
    //country = country.capitalizedString;
    if ([inValue isEqualToString:@"10"]) {
        return NO;
    }
    return YES;
}


@end



    Person * personOne =[[Person alloc] init];
    personOne.age=13;
    NSError* error;
    id value = @"10";
    NSString* key = @"age";
    BOOL result = [personOne validateValue:&value forKey:key error:&error]; //如果没有重写-(BOOL)-validate<Key>:error:,默认返回Yes
    if (result) {
        NSLog(@"键值匹配");
        [personOne setValue:value forKey:key];
    }
    else{
        NSLog(@"键值不匹配"); //不能设为日本,基他国家都行
    }
    NSString* age = [personOne valueForKey:@"age"];
    NSLog(@"age:%@",age);
/*
2017-08-29 14:31:27.850 KVC[4580:93769] 键值不匹配
2017-08-29 14:31:27.850 KVC[4580:93769] age:13
*/
  • Accessor Search Patterns(搜索key的模式)
  • Setter的基本搜索方式

规则也是: 先找相关方法,再找相关变量
1.先去找相关方法 :set<Key>
2.那么去判断类的实现文件中有没有实现:+(BOOL)accessInstanceVariablesDirectly,不实现默认返回YES
3.如果返回no,则执行setValue:(id)value forUndefinedKey:(NSString *)key(抛出一个异常,如果实现文件中没有重写该方法,则会奔溃,如果重写了,则不会奔溃,还可以在这个方法给没有找到的key一个替代值)
4.如果返回的yes,则找相关变量:那么按_<key>,_is<Key>,<key>,is<key>的顺序搜索成员名,如果也没找到则执行setValue:(id)value forUndefinedKey:(NSString *)key

  • Getter的基本搜索方式

总体规则是:先找相关方法,在找相关
1.先去找相关方法,如果相关方法没找到
2.那么去判断类的实现文件中有没有实现:+(BOOL)accessInstanceVariablesDirectly,不实现默认返回YES,
3.如果返回NO,直接执行kvc的:setValue:(id)value forUndefinedKey:(NSString *)key方法(抛出一个异常,如果实现文件中没有重写该方法,则会奔溃,如果重写了,则不会奔溃,还可以在这个方法给没有找到的key一个替代值)
4.如果返回的是yes,则继续再去找相关变量
5.如果还没找到相关,也会执行:setValue:(id)value forUndefinedKey:(NSString *)key方法

相关方法指的是:
1.get<Key>,<key>,is<Key>,或者_<key>(官网上说了还可以有这个方法,但是我试了并没有用,那么就先保险的确定前三个吧)
2.如果上面的getter没有找到,KVC则会查找countOf<Key>,objectIn<Key>AtIndex,<Key>AtIndex格式的方法。如果countOf<Key>和另外两个方法中的要个被找到,那么就会返回一个可以响应NSArray所的方法的代理集合(它是NSKeyValueArray,是NSArray的子类),调用这个代理集合的方法,或者说给这个代理集合发送NSArray的方法,就会以countOf<Key>,objectIn<Key>AtIndex,<Key>AtIndex这几个方法组合的形式调用。还有一个可选的get<Ket>:range:方法。所以你想重新定义KVC的一些功能,你可以添加这些方法,需要注意的是你的方法名要符合KVC的标准命名方法,包括方法签名
3.还没查到,那么查找countOf<Key>、enumeratorOf<Key>、memberOf<Key>:格式的方法。
如果这三个方法都找到,那么就返回一个可以响应NSSet所有方法的代理集合(collection proxy object)。发送给这个代理集合(collection proxy object)的NSSet消息方法,就会以countOf<Key>、enumeratorOf<Key>、memberOf<Key>:组合的形式调用

相关变量:
那么按_<key>,_is<Key>,<key>,is<key>的顺序直接搜索成员名
这里不推荐这么做,因为这样直接访问实例变量破坏了封装性

  • Getter的有序集合的搜索方式(比如NSMutableArray)

mutableArrayValueForKey:搜索方式如下:

  1. 搜索insertObject:in<Key>AtIndex:、removeObjectFrom<Key>AtIndex:或者insert<Key>:atIndexes、remove<Key>AtIndexes:格式的方法。
    如果至少一个insert方法和至少一个remove方法找到,那么同样返回一个可以响应NSMutableArray所有方法的代理集合。那么发送给这个代理集合的NSMutableArray消息方法,以insertObject:in<Key>AtIndex:、removeObjectFrom<Key>AtIndex:、insert<Key>:atIndexes、remove<Key>AtIndexes:组合的形式调用。还有两个可选实现的接口:replaceObjectIn<Key>AtIndex:withObject:、replace<Key>AtIndexes:with<Key>:。
  2. 否则,搜索set<Key>:格式的方法,如果找到,那么发送给代理集合的NSMutableArray最终都会调用set<Key>:方法。
    也就是说,mutableArrayValueForKey取出的代理集合修改后,用set<Key>:重新赋值回去。这样做效率会差很多,所以推荐实现上面的方法。
  3. 否则,那么如果类方法accessInstanceVariablesDirectly返回YES,那么按_<key>,<key>的顺序直接搜索成员名。如果找到,那么发送的NSMutableArray消息方法直接转交给这个成员处理。
  4. 再找不到,调用setValue:forUndefinedKey:
  • Getter的搜索无序集合成员,如:NSSet

mutableSetValueForKey:搜索方式如下:

  1. 搜索add<Key>Object:、remove<Key>Object:或者add<Key>:、remove<Key>:格式的方法,如果至少一个insert方法和至少一个remove方法找到,那么返回一个可以响应NSMutableSet所有方法的代理集合。那么发送给这个代理集合的NSMutableSet消息方法,以add<Key>Object:、remove<Key>Object:、add<Key>:、remove<Key>:组合的形式调用。还有两个可选实现的接口:intersect<Key>、set<Key>:。
  2. 如果reciever是ManagedObejct,那么就不会继续搜索了。
  3. 否则,搜索set<Key>:格式的方法,如果找到,那么发送给代理集合的NSMutableSet最终都会调用set<Key>:方法。也就是说,mutableSetValueForKey取出的代理集合修改后,用set<Key>:重新赋值回去。这样做效率会差很多,所以推荐实现上面的方法。
  4. 否则,那么如果类方法accessInstanceVariablesDirectly返回YES,那么按_<key>,<key>的顺序直接搜索成员名。如果找到,那么发送的NSMutableSet消息方法直接转交给这个成员处理。
  5. 再找不到,调用setValue:forUndefinedKey:。

关于Getter的搜索比如mutableArrayValueForKey和
mutableSetValueForKey我几乎没有用过,如果有人用的很溜,请指教

  • 总结

相比直接访问KVC的效率会稍低一点,所以只有当你非常需要它提供的可扩展性时才使用它。

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

推荐阅读更多精彩内容