Objective-C KVC机制深入理解

1. 基本概念

MODEL


主要是英文文档里面经常出现的一些概念,讲解一下,方便英文文档的阅读。


iOS应用开发是遵循MVC设计模式的,Cocoa框架用Object Modeling的规则来规范一个Model的实现。


ObjectModeling有如下几个概念的规定:


Entity:表示持有数据的一个实体


Property实体中的成员,分为Attribute和:Relationship


Attribute:基本类型的成员,比如:数字、NSString。


Relationship:指向其它Entity的关系型成员,它又有to 1Relationship和to manyRelationship的区别。


AccessorMethod:getter,setter。


举例:


如下是一个部门和员工关系的Model


部门:Department


部门名称(NSString)


成员(NSArray)


部长(Employee)


MIC


(所有成员)


老王(一个成员)


MIB


员工:Employee


名字(NSStirng)


所属部门(Department)


小王


MIC



使用KVC、KVO的优势


通过规定了一组通用的Cocoa命名法则、调用规则等,实现了如下功能:


² 使用一对高度规范化的访问方法,获取以及设置任何对象的任何属性的值。


² 通过继承一个特定的方法,并且指定希望监视的对象及希望监视的属性名称,就能在该对象的指定属性的值发生改变时,得到一个“通知”(尽管这不是一个真正意 义上的通知),并且得到相关属性的值的变化(原先的值和改变后的新值)。


² 通过一个简单的函数调用,使一个视图对象的一个指定属性随时随地都和一个控制器对象或模型对象的一个指定属性保持同步。


2. KVC


2.1 概述


KVC是KeyValue Coding的简称,它是一种可以直接通过字符串的名字(key)来访问类属性的机制。而不是通过调用Setter、Getter方法访问。


当使用KVO、Core Data、CocoaBindings、AppleScript(Mac支持)时,KVC是关键技术。


2.2 如何使用KVC


关键方法定义在:NSKeyValueCodingprotocol


KVC支持类对象和内建基本数据类型。


2.2.1 获取值


valueForKey:,传入NSString属性的名字。


valueForKeyPath:,传入NSString属性的路径,xx.xx形式。


valueForUndefinedKey它的默认实现是抛出异常,可以重写这个函数做错误处理。


2.2.2 修改值


setValue:forKey:


setValue:forKeyPath:


setValue:forUndefinedKey:


setNilValueForKey: 当对非类对象属性设置nil时,调用,默认抛出异常。


2.2.3 一对多关系成员的情况


mutableArrayValueForKey:有序一对多关系成员 NSArray


mutableSetValueForKey:无序一对多关系成员 NSSet


示例:


2.3 KVC的实现细节


搜索Setter、Getter方法


 这一部分比较重要,能让你了解到KVC调用之后,到底是怎样获取和设置类成员值的。


补充一点个人理解:KVC传入的KEY为字符串,字符串转换为OC下的类的属性的过程基于反射机制;


2.3.1 搜索简单的成员


如:基本类型成员,单个对象类型成员:NSInteger,NSString*成员。


a. setValue:forKey的搜索方式:


1. 首先搜索set:方法


如果成员用@property,@synthsize处理,因为@synthsize告诉编译器自动生成set:格式的setter方法,所以这种情况下会直接搜索到。


注意:这里的是指成员名,而且首字母大写。下同。


2. 上面的setter方法没有找到,如果类方法accessInstanceVariablesDirectly返回YES(注:这是NSKeyValueCodingCatogery中实现的类方法,默认实现为返回YES)。


那么按_,_is,,is的顺序搜索成员名。


3. 如果找到设置成员的值,如果没有调用setValue:forUndefinedKey:。


b. valueForKey:的搜索方式:


1. 首先按get、、is的顺序查找getter方法,找到直接调用。如果是bool、int等内建值类型,会做NSNumber的转换。


2. 上面的getter没有找到,查找countOf、objectInAtIndex:、AtIndexes格式的方法。


如果countOf和另外两个方法中的一个找到,那么就会返回一个可以响应NSArray所有方法的代理集合(collection proxy object)。发送给这个代理集合(collection proxy object)的NSArray消息方法,就会以countOf、objectInAtIndex:、AtIndexes这几个方法组合的形式调用。还有一个可选的get:range:方法。


3. 还没查到,那么查找countOf、enumeratorOf、memberOf:格式的方法。


如果这三个方法都找到,那么就返回一个可以响应NSSet所有方法的代理集合(collection proxy object)。发送给这个代理集合(collection proxy object)的NSSet消息方法,就会以countOf、enumeratorOf、memberOf:组合的形式调用。


4. 还是没查到,那么如果类方法accessInstanceVariablesDirectly返回YES,那么按_,_is,,is的顺序直接搜索成员名。


5. 再没查到,调用valueForUndefinedKey:。


2.3.2 查找有序集合成员,比如NSMutableArray


mutableArrayValueForKey:搜索方式如下:


1. 搜索insertObject:inAtIndex:、removeObjectFromAtIndex:或者insert:atIndexes、removeAtIndexes:格式的方法。


如果至少一个insert方法和至少一个remove方法找到,那么同样返回一个可以响应NSMutableArray所有方法的代理集合。那么发送给这个代理集合的NSMutableArray消息方法,以insertObject:inAtIndex:、removeObjectFromAtIndex:、insert:atIndexes、removeAtIndexes:组合的形式调用。还有两个可选实现的接口:replaceObjectInAtIndex:withObject:、replaceAtIndexes:with:。


2. 否则,搜索set:格式的方法,如果找到,那么发送给代理集合的NSMutableArray最终都会调用set:方法。


也就是说,mutableArrayValueForKey取出的代理集合修改后,用set:重新赋值回去。这样做效率会差很多,所以推荐实现上面的方法。


3. 否则,那么如果类方法accessInstanceVariablesDirectly返回YES,那么按_,的顺序直接搜索成员名。如果找到,那么发送的NSMutableArray消息方法直接转交给这个成员处理。


4. 再找不到,调用setValue:forUndefinedKey:。


2.3.3 搜索无序集合成员,如:NSSet。


mutableSetValueForKey:搜索方式如下:


1. 搜索addObject:、removeObject:或者add:、remove:格式的方法,如果至少一个insert方法和至少一个remove方法找到,那么返回一个可以响应NSMutableSet所有方法的代理集合。那么发送给这个代理集合的NSMutableSet消息方法,以addObject:、removeObject:、add:、remove:组合的形式调用。还有两个可选实现的接口:intersect、set:。


2. 如果reciever是ManagedObejct,那么就不会继续搜索了。


3. 否则,搜索set:格式的方法,如果找到,那么发送给代理集合的NSMutableSet最终都会调用set:方法。也就是说,mutableSetValueForKey取出的代理集合修改后,用set:重新赋值回去。这样做效率会差很多,所以推荐实现上面的方法。


4. 否则,那么如果类方法accessInstanceVariablesDirectly返回YES,那么按_,的顺序直接搜索成员名。如果找到,那么发送的NSMutableSet消息方法直接转交给这个成员处理。


5. 再找不到,调用setValue:forUndefinedKey:。


KVC还提供了下面的功能


2.4 值的正确性核查


KVC提供属性值确认的API,它可以用来检查set的值是否正确、为不正确的值做一个替换值或者拒绝设置新值并返回错误原因。


实现核查方法


为如下格式:validate:error:


如:


[cpp] view plain copy

print?

-(BOOL)validateName:(id *)ioValue error:(NSError **)outError

{

// The name must not be nil, and must be at least two characters long.

if ((*ioValue == nil) || ([(NSString *)*ioValue length] < 2]) {

if (outError != NULL) {

NSString *errorString = NSLocalizedStringFromTable(

@"A Person's name must be at least two characters long", @"Person",

@"validation: too short name error");

NSDictionary *userInfoDict =

[NSDictionary dictionaryWithObject:errorString

forKey:NSLocalizedDescriptionKey];

*outError = [[[NSError alloc] initWithDomain:PERSON_ERROR_DOMAIN

code:PERSON_INVALID_NAME_CODE

userInfo:userInfoDict] autorelease];

}

return NO;

}

return YES;

}


-(BOOL)validateName:(id *)ioValue error:(NSError **)outError

{

   // The name must not be nil, and must be at least two characters long.

   if ((*ioValue == nil) || ([(NSString *)*ioValue length] < 2]) {

       if (outError != NULL) {

           NSString *errorString = NSLocalizedStringFromTable(

                   @"A Person's name must be at least two characters long", @"Person",

                   @"validation: too short name error");

           NSDictionary *userInfoDict =

               [NSDictionary dictionaryWithObject:errorString

                                           forKey:NSLocalizedDescriptionKey];

           *outError = [[[NSError alloc] initWithDomain:PERSON_ERROR_DOMAIN

                                                   code:PERSON_INVALID_NAME_CODE

                                               userInfo:userInfoDict] autorelease];

       }

       return NO;

   }

   return YES;

}

调用核查方法:


validateValue:forKey:error:,默认实现会搜索 validate:error:格式的核查方法,找到则调用,未找到默认返回YES。


注意其中的内存管理问题。


2.5 集合操作


集合操作通过对valueForKeyPath:传递参数来使用,一定要用在集合(如:array)上,否则产生运行时刻错误。其格式如下:


Left keypath部分:需要操作对象路径。


Collectionoperator部分:通过@符号确定使用的集合操作。


Rightkey path部分:需要进行集合操作的属性。


2.5.1 数据操作


@avg:平均值


@count:总数


@max:最大


@min:最小


@sum:总数


确保操作的属性为数字类型,否则运行时刻错误。


2.5.2 对象操作


针对数组的情况


@distinctUnionOfObjects:返回指定属性去重后的值的数组


@unionOfObjects:返回指定属性的值的数组,不去重


属性的值不能为空,否则产生异常。


2.5.3 数组操作


针对数组的数组情况


@distinctUnionOfArrays:返回指定属性去重后的值的数组


@unionOfArrays:返回指定属性的值的数组,不去重


@distinctUnionOfSets:同上,只是返回值为NSSet


示例代码:


2.6 效率问题


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

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

推荐阅读更多精彩内容