理解KVC与KVO

理解KVC与KVO


KVC(Key-Value Coding)和KVO(Key-Value Observing)可能是Objective-C中最强大最实用的功能之一。

下面的几个例子可以帮你更好的理解它们。

原文作者:Marcus Zarra

原文地址

当通过KVC调用对象时,比如:[self valueForKey:@”someKey”]时,程序会自动试图通过几种不同的方式解析这个调用。首先查找对象是否带有 someKey 这个方法,如果没找到,会继续查找对象是否带有someKey这个实例变量(iVar),如果还没有找到,程序会继续试图调用 -(id) valueForUndefinedKey:这个方法。如果这个方法还是没有被实现的话,程序会抛出一个NSUndefinedKeyException异常错误。

(cocoachina.com注:Key-Value Coding查找方法的时候,不仅仅会查找someKey这个方法,还会查找getsomeKey这个方法,前面加一个get,或者_someKey以及_getsomeKey这几种形式。同时,查找实例变量的时候也会不仅仅查找someKey这个变量,也会查找_someKey这个变量是否存在。)

设计valueForUndefinedKey:方法的主要目的是当你使用-(id)valueForKey方法从对象中请求值时,对象能够在错误发生前,有最后的机会响应这个请求。这样做有很多好处,下面的两个例子说明了这样做的好处。

Core Data参数

在其中一个方法中,我做了一个单独的对象,来响应参数请求。这样程序的任何部分要调用参数时,无需每次都在需要时必须创建NSfetchRequest,而是直接通过一小段干净的代码直接获取。这样一来,程序的其他部分只需要这样简单地进行调用:

NSString *companyName = [[Company shared]valueForKey:kCompanyName];

而无需使用下面这么多代码:

NSString *companyName =nil;

NSFetchRequest *request = [[[NSFetchRequestalloc]init]autorelease];

[requestsetEntity:[NSEntityDescriptionentityForName:@"Parameter"inManagedObjectContext:managedObjectContext]]];

[requestsetPredicate:[NSPredicatepredicateWithFormat:@"name == %@", kCompanyName]];

NSError *error =nil;

companyName = [[managedObjectContextexecuteFetchRequest:requesterror:&error]lastObject];

NSAssert(error == nil, ([NSStringstringWithFormat:@"Error requesting parameter: %@\n%@", kCompanyName, error]));


最糟糕的情况是当我想设置一个参数值时:

id parameter;

NSFetchRequest *request = [[[NSFetchRequestalloc]init]autorelease];

[requestsetEntity:[NSEntityDescriptionentityForName:@"Parameter"inManagedObjectContext:managedObjectContext]];

[requestsetPredicate:[NSPredicatepredicateWithFormat:@"name == %@",key]];

parameter = [[managedObjectContextexecuteFetchRequest:requesterror:&error]lastObject];

NSAssert(error == nil, ([NSStringstringWithFormat:@"Error getting parameter %@:%@", key,error]));

if (!parameter) {

parameter = [NSEntityDescriptioninsertNewObjectForEntityForName:@"Parameter"inManagedObjectContext:managedObjectContext];

[parametersetValue:keyforKey:@"name"];

}

if ([valueisKindOfClass:[NSNumberclass]]) {

[parametersetValue:[valuestringValue]forKey:@"value"];

}elseif ([valueisKindOfClass:[NSDateclass]]) {

[parametersetValue:[valuedescription]forKey:@"value"];

}else {

[parametersetValue:valueforKey:@"value"];

}


你可以看出来,在第5或者第6次插入这段代码,仅仅为了从Core Data存储中获取一个小小的参数。

要注意的是这些参数和NSUserDefaults不同,NSUserDefaults用来保存应用程序级的参数是很棒的,但是如果你需要保存文档级的参数,就不能用NSUserDefaults了。

为了不用写大量代码来存取Core Data堆栈,参数可以集中放置在单独的对象中。这个对象可以面向程序的任何部分。然后所需的参数通过valueForUndefinedKey:方法直接读取或者通过setValue:forUndefinedKey:方法直接设置,看一下下面的例子:

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

{

NSFetchRequest *request = [[[NSFetchRequestalloc]init]autorelease];

[requestsetEntity:[NSEntityDescriptionentityForName:@"Parameter"inManagedObjectContext:managedObjectContext]]];

[requestsetPredicate:[NSPredicatepredicateWithFormat:@"name == %@",key]];

NSError *error =nil;

NSManagedObject *parameter = [[managedObjectContextexecuteFetchRequest:requesterror:&error]lastObject];

NSAssert(error ==nil, ([NSStringstringWithFormat:@"Error requesting parameter: %@\n%@", key, error]));

return [parametervalueForKey:@"value"];

}

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

{

[selfwillChangeValueForKey:key];

NSFetchRequest *request = [[[NSFetchRequestalloc]init]autorelease];

[requestsetEntity:[NSEntityDescriptionentityForName:@"Parameter"inManagedObjectContext:managedObjectContext]];

[requestsetPredicate:[NSPredicatepredicateWithFormat:@"name == %@",key]];

NSError *error =nil;

NSManagedObject *parameter = [[managedObjectContextexecuteFetchRequest:requesterror:&error]lastObject];

NSAssert(error ==nil, ([NSStringstringWithFormat:@"Error getting parameter %@:%@", key,error]));

if (!parameter) {

parameter = [NSEntityDescriptioninsertNewObjectForEntityForName:@"Parameter"inManagedObjectContext:managedObjectContext];

[parametersetValue:keyforKey:@"name"];

}

if ([valueisKindOfClass:[NSNumberclass]]) {

[parametersetValue:[valuestringValue]forKey:@"value"];

}elseif ([valueisKindOfClass:[NSDateclass]]) {

[parametersetValue:[valuedescription]forKey:@"value"];

}else {

[parametersetValue:valueforKey:@"value"];

}

[selfdidChangeValueForKey:key];

}

你可以看出,这段代码非常直接。不过代码中仅有一个地方在整个程序中没有被丢掉。另外,如果我希望修改存储参数的位置,我只需修改一次即可。

通过使用valueForUndefinedKey:方法,我可以避免每次希望保存参数时都修改代码。最后,由于参数的键被认为是在单独对象中的值,我可以在它改变的时候观察并接收到通知,这样我可以将Interface Builder中的绑定直接用到这个对象上。

(待续)

另外,如果你希望了解更多关于Key-Value Coding方面的内容,可以看看Allen Dang的这篇文章:Key-Value Coding详解。http://blog.codingmylife.com/?p=15 他的Blog改版的,挺漂亮的:)


  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值