KVC与KVO
来源:互联网 发布:股票分时数据 编辑:程序博客网 时间:2024/04/29 05:29
原文:http://www.cocoachina.com/macdev/cocoa/2009/0611/221.html
KVC(Key-Value Coding)和KVO(Key-Value Observing)可能是Objective-C中最强大最实用的功能之一。
当通过KVC调用对象时,比如:[self valueForKey:@”someKey”]时,程序会自动试图通过几种不同的方式解析这个调用。首先查找对象是否带有 someKey 这个方法,如果没找到,会继续查找对象是否带有someKey这个实例变量(iVar),如果还没有找到,程序会继续试图调用 -(id) valueForUndefinedKey:这个方法。如果这个方法还是没有被实现的话,程序会抛出一个NSUndefinedKeyException异常错误。
设计valueForUndefinedKey:方法的主要目的是当你使用-(id)valueForKey方法从对象中请求值时,对象能够在错误发生前,有最后的机会响应这个请求。这样做有很多好处,下面的两个例子说明了这样做的好处。
Core Data参数
在其中一个方法中,我做了一个单独的对象,来响应参数请求。这样程序的任何部分要调用参数时,无需每次都在需要时必须创建NSfetchRequest,而是直接通过一小段干净的代码直接获取。这样一来,程序的其他部分只需要这样简单地进行调用:
NSString *companyName = [[Company shared] valueForKey:kCompanyName];
而无需使用下面这么多代码:
NSString *companyName = nil;NSFetchRequest *request = [[[NSFetchRequest alloc] init] autorelease];[request setEntity:[NSEntityDescription entityForName:@"Parameter"inManagedObjectContext:managedObjectContext]]];[request setPredicate:[NSPredicate predicateWithFormat:@"name == %@", kCompanyName]];NSError *error = nil;companyName = [[managedObjectContext executeFetchRequest:request error:&error] lastObject];NSAssert(error == nil, ([NSString stringWithFormat:@"Error requesting parameter: %@\n%@", kCompanyName,error]));
最糟糕的情况是当我想设置一个参数值时:
id parameter;NSFetchRequest *request = [[[NSFetchRequest alloc] init] autorelease];[request setEntity:[NSEntityDescription entityForName:@"Parameter"inManagedObjectContext:managedObjectContext]];[request setPredicate:[NSPredicate predicateWithFormat:@"name == %@", key]];parameter = [[managedObjectContext executeFetchRequest:request error:&error] lastObject];NSAssert(error == nil, ([NSString stringWithFormat:@"Error getting parameter %@:%@", key, error]));if (!parameter) { parameter = [NSEntityDescription insertNewObjectForEntityForName:@"Parameter"inManagedObjectContext:managedObjectContext]; [parameter setValue:key forKey:@"name"];}if ([value isKindOfClass:[NSNumber class]]) { [parameter setValue:[value stringValue] forKey:@"value"];} else if ([value isKindOfClass:[NSDate class]]) { [parameter setValue:[value description] forKey:@"value"];} else { [parameter setValue:value forKey:@"value"];}
你可以看出来,在第5或者第6次插入这段代码,仅仅为了从Core Data存储中获取一个小小的参数。
要注意的是这些参数和NSUserDefaults不同,NSUserDefaults用来保存应用程序级的参数是很棒的,但是如果你需要保存文档级的参数,就不能用NSUserDefaults了。
为了不用写大量代码来存取Core Data堆栈,参数可以集中放置在单独的对象中。这个对象可以面向程序的任何部分。然后所需的参数通过valueForUndefinedKey:方法直接读取或者通过setValue:forUndefinedKey:方法直接设置,看一下下面的例子
- (id)valueForUndefinedKey:(NSString *)key;{ NSFetchRequest *request = [[[NSFetchRequest alloc] init] autorelease]; [request setEntity:[NSEntityDescription entityForName:@"Parameter"inManagedObjectContext:managedObjectContext]]]; [request setPredicate:[NSPredicate predicateWithFormat:@"name == %@", key]]; NSError *error = nil; NSManagedObject *parameter = [[managedObjectContext executeFetchRequest:request error:&error] lastObject]; NSAssert(error == nil, ([NSString stringWithFormat:@"Error requesting parameter: %@\n%@", key, error])); return [parameter valueForKey:@"value"];} - (void)setValue:(id)value forUndefinedKey:(NSString *)key { [self willChangeValueForKey:key]; NSFetchRequest *request = [[[NSFetchRequest alloc] init] autorelease]; [request setEntity:[NSEntityDescription entityForName:@"Parameter"inManagedObjectContext:managedObjectContext]]; [request setPredicate:[NSPredicate predicateWithFormat:@"name == %@", key]]; NSError *error = nil; NSManagedObject *parameter = [[managedObjectContext executeFetchRequest:request error:&error] lastObject]; NSAssert(error == nil, ([NSString stringWithFormat:@"Error getting parameter %@:%@" , key, error])); if (!parameter) { parameter = [NSEntityDescription insertNewObjectForEntityForName:@"Parameter"inManagedObjectContext:managedObjectContext]; [parameter setValue:key forKey:@"name"]; } if ([value isKindOfClass:[NSNumber class]]) { [parameter setValue:[value stringValue] forKey:@"value"]; } else if ([value isKindOfClass:[NSDate class]]) { [parameter setValue:[value description] forKey:@"value"]; } else { [parameter setValue:value forKey:@"value"]; } [self didChangeValueForKey:key];}
你可以看出,这段代码非常直接。不过代码中仅有一个地方在整个程序中没有被丢掉。另外,如果我希望修改存储参数的位置,我只需修改一次即可。
通过使用valueForUndefinedKey:方法,我可以避免每次希望保存参数时都修改代码。最后,由于参数的键被认为是在单独对象中的值,我可以在它改变的时候观察并接收到通知,这样我可以将Interface Builder中的绑定直接用到这个对象上
- 理解KVC与KVO
- KVC 与 KVO 理解
- KVC 与 KVO 理解
- KVC 与 KVO 理解
- KVC与KVO
- KVC 与 KVO 理解
- KVC与KVO解析
- kvc与kvo
- KVC与KVO:
- KVC 与 KVO 理解
- KVC 与 KVO 理解
- 【iOS】KVC 与 KVO
- KVC 与 KVO 理解
- KVC与KVO理解
- KVC 与 KVO 理解
- KVC 与 KVO 理解
- KVC 与 KVO 理解
- KVC与KVO
- c++的引用(未完成)
- 设计模式之 合成聚合复用原则
- Android 环境Bug解决办法
- Cannot convert value '0000-00-00 00:00:00' from column 1 to TIMESTAMP
- Codeforces Round #194 (Div. 2) / 334A Candy Bags(模拟)
- KVC与KVO
- 图片对象转换工具
- 一、什么是OpenGL
- Oracle并行服务器(OPS)就是RAC的前身
- VC中操作excel表格 ListCtrl数据导出到Excel
- 如何设计好部落格的十个技巧
- linux中mount命令的基本用法
- 排序算法之冒泡排序
- 微信公众平台开发[5]-开发模式-高级功能-获取access_token