kvc在我的脑海里用来更改属性的实例变量值。

今天,他们遇到了kvc第二次去学习它,在网上看了很多博客,这似乎不符合我的口味,为了提取一些以下的。总结自己的。

http://www.cnblogs.com/stoic/archive/2012/07/20/2601315.html

这个博主写的是一些应用实例。我比較喜欢,他说明了。怎样去代码操作;

http://blog.csdn.net/omegayy/article/details/7381301

这个博主是基本的原理。

以下是自己感觉实用的东西从两位博客中摘录的。

概述

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

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

怎样使用KVC

关键方法定义在:NSKeyValueCodingprotocol

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

       获取值

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

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

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

        改动值

setValue:forKey:

setValue:forKeyPath:

setValue:forUndefinedKey:

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

       一对多关系成员的情况

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

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

以下是一些操作实例:

1、使用KVC

#import <Foundation/Foundation.h>

@interface Student : NSObject
{
NSString *name;
}
@end
#import "Student.h" int main(int argc, const char * argv[])
{
@autoreleasepool {
Student *student = [[[Student alloc]init ]autorelease];
[student setValue:@"张三" forKey:@"name"];
NSString *name = [student valueForKey:@"name"];
NSLog(@"学生姓名:%@",name);
}
return 0;
}

2、键路径訪问属性

#import <Foundation/Foundation.h>

@interface Course : NSObject
{
NSString *CourseName;
}
@end
#import "Course.h" @implementation Course @end
#import <Foundation/Foundation.h>
@class Course;
@interface Student : NSObject
{
NSString *name;
Course *course;
}
@end
#import "Student.h"
#import "Course.h" int main(int argc, const char * argv[])
{
@autoreleasepool {
Student *student = [[[Student alloc]init ]autorelease];
[student setValue:@"张三" forKey:@"name"];
NSString *name = [student valueForKey:@"name"];
NSLog(@"学生姓名:%@",name); Course *course = [[[Course alloc]init] autorelease];
[course setValue:@"语文课" forKey:@"CourseName"];
[student setValue:course forKey:@"course"];
NSString *courseName = [student valueForKeyPath:@"course.CourseName"];
NSLog(@"课程名称:%@", courseName); //也能够这样存值
[student setValue:@"数学课" forKeyPath:@"course.CourseName"];
courseName = [student valueForKeyPath:@"course.CourseName"];
NSLog(@"课程名称:%@", courseName); }
return 0;
}

3、自己主动封装基本数据类型

#import <Foundation/Foundation.h>
@class Course;
@interface Student : NSObject
{
NSString *name;
Course *course;
NSInteger point;
}
@end
#import "Student.h"
#import "Course.h" int main(int argc, const char * argv[])
{
@autoreleasepool {
Student *student = [[[Student alloc]init ]autorelease];
[student setValue:@"张三" forKey:@"name"];
NSString *name = [student valueForKey:@"name"];
NSLog(@"学生姓名:%@",name); Course *course = [[[Course alloc]init] autorelease];
[course setValue:@"语文课" forKey:@"CourseName"];
[student setValue:course forKey:@"course"];
NSString *courseName = [student valueForKeyPath:@"course.CourseName"];
NSLog(@"课程名称:%@", courseName); //也能够这样存值
[student setValue:@"数学课" forKeyPath:@"course.CourseName"];
courseName = [student valueForKeyPath:@"course.CourseName"];
NSLog(@"课程名称:%@", courseName); [student setValue:@"88" forKeyPath:@"point"];
NSString *point = [student valueForKey:@"point"];
NSLog(@"分数:%@", point); }
return 0;
}

4、操作集合

#import <Foundation/Foundation.h>
@class Course;
@interface Student : NSObject
{
NSString *name;
Course *course;
NSInteger point;
NSArray *otherStudent;
}
@end
#import "Student.h"
#import "Course.h" int main(int argc, const char * argv[])
{
@autoreleasepool {
Student *student = [[[Student alloc]init ]autorelease];
[student setValue:@"张三" forKey:@"name"];
NSString *name = [student valueForKey:@"name"];
NSLog(@"学生姓名:%@",name); [student setValue:@"88" forKey:@"point"];
NSString *point = [student valueForKey:@"point"];
NSLog(@"分数:%@", point); Student *student1 = [[[Student alloc]init]autorelease];
Student *student2 = [[[Student alloc]init]autorelease];
Student *student3 = [[[Student alloc]init]autorelease];
[student1 setValue:@"65" forKey:@"point"];
[student2 setValue:@"77" forKey:@"point"];
[student3 setValue:@"99" forKey:@"point"];
NSArray *array = [NSArray arrayWithObjects:student1,student2,student3,nil];
[student setValue:array forKey:@"otherStudent"];
NSLog(@"其它学生的成绩%@", [student valueForKeyPath:@"otherStudent.point"]);
NSLog(@"共%@个学生", [student valueForKeyPath:@"otherStudent.@count"]);
NSLog(@"最高成绩:%@", [student valueForKeyPath:@"otherStudent.@max.point"]);
NSLog(@"最低成绩:%@", [student valueForKeyPath:@"otherStudent.@min.point"]);
NSLog(@"平均成绩:%@", [student valueForKeyPath:@"otherStudent.@avg.point"]);
}
return 0;
}

KVC的实现细节:

搜索Setter、Getter方法

 这一部分比較重要,能让你了解到KVC调用之后。究竟是如何获取和设置类成员值的。

搜索简单的成员

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

a. setValue:forKey的搜索方式:

1. 首先查找设置改属性的setter方法

假设成员用@property,@synthsize处理。由于@synthsize告诉编译器自己主动生成setter:格式的setter方法,所以这样的情况下会直接搜索到。

注意:setting方法是泛指 ,指为属性赋值的方法。

2. 上面的setter方法没有找到,假设类方法accessInstanceVariablesDirectly(该方法会返回是否直接訪问没有生成成訪问器的变量)返回YES(注:这是NSKeyValueCodingCatogery中实现的类方法,默认实现为返回YES)。

那么按_name,_isnamename。is

name的顺序搜索成员名。(设变量名字为name

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

b. valueForKey:的搜索方式:

设变量名字为name

1. 首先按getnamename、isname的顺序查找getter方法。找到直接调用。

假设是bool、int等内建值类型,会做NSNumber的转换。

2. 上面的getter没有找到。查找countOfname、objectInnameAtIndex:、nameAtIndexes格式的方法。

假设countOf<Key>和另外两个方法中的一个找到,那么就会返回一个能够响应NSArray全部方法的代理集合(collection proxy object)。

发送给这个代理集合(collection proxy object)的NSArray消息方法,就会以countOfname、objectInnameAtIndex:、nameAtIndexes这几个方法组合的形式调用。另一个可选的getname:range:方法。

3. 还没查到,那么查找countOfname、enumeratorOfname、memberOfname:格式的方法。

假设这三个方法都找到,那么就返回一个能够响应NSSet全部方法的代理集合(collection proxy object)。发送给这个代理集合(collection proxy object)的NSSet消息方法,就会以countOfname、enumeratorOfname、memberOfname:组合的形式调用。

4. 还是没查到。那么假设类方法accessInstanceVariablesDirectly返回YES。那么按_name。_isnamename,isname的顺序直接搜索成员名。

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

2.3.2         查找有序集合成员,比方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:。

 NSMutableArray *mutablearray = [kvo mutableArrayValueForKey:@"array"];//kvo为自己定义的类里面放了一个array的数组通过这种方法来获取该数组的代理,并在此改动。kvo中的array值就会对应的变化;
[mutablearray removeObject:@"234"];
[mutablearray addObject:@"345"];

2.3.3         搜索无序集合成员。如: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:。

KVC还提供了以下的功能

2.4  值的正确性核查

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

实现核查方法

为例如以下格式:validate<Key>:error:

如:

-(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<Key>: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的效率会稍低一点,所以仅仅有当你很须要它提供的可扩展性时才使用它。

以下摘录一段留言能让我们了解为什么会用kvc:

. KVO是以KVC为基础的。有些监听不使用KVC操作会监听不到。

2. 假设用到Core Data。须要使用KVC做值的存取。(否则你会发现各种各样的问题的)

主要应用场景就是与KVO和Core Data协作。

这样的统一的直接通过字符串存取ObjC中对象的成员属性的接口,能够实现由外部脚本控件程序运行或者获取程序运行信息。

通过KVC存取二进制库中的私有成员也比較有用。

事实上,平均开发者,并不需要用太多。

版权声明:本文博客原创文章,博客,未经同意,不得转载。

iOS kvc的更多相关文章

  1. iOS KVC详细讲解

    iOS KVC详细讲解 什么是KVC? KVC即NSKeyValueCoding,就是键-值编码的意思.一个非正式的 Protocol,是一种间接访问对象的属性使用字符串来标识属性,而不是通过调用存取 ...

  2. iOS - KVC 键值编码

    1.KVC KVC 是 Key-Value Coding 的简写,是键值编码的意思,属于 runtime 方法.Key Value Coding 是 cocoa 的一个标准组成部分,是间接给对象属性设 ...

  3. [iOS] KVC 和 KVO

    开发iOS经常会看见KVO和KVC这两个概念,特地了解了一下. 我的新博客wossoneri.com link KVC Key Value Coding KVC是一种用间接方式访问类的属性的机制.比如 ...

  4. iOS:KVC和KVO

    来源:  对月流 链接:http://www.jianshu.com/p/f1393d10109d 写在前面: 关于KVC和KVO各种博客多了去了,重新整理下,就当是温习一下吧,也还算是个新手,不对的 ...

  5. iOS KVC,KVO

    链接(写得不错,着重kvc):http://www.cocoachina.com/industry/20140224/7866.html 链接:http://www.cnblogs.com/kensh ...

  6. iOS KVC/KVO/KVB

    看了那么多博客.描述那么复杂,其实KVC很简单,没描述的那么复杂,所以写一篇简单的易于理解的博文,切入正文: 1.KVC底层是通过runtime对method和value操作  比如说如下的一行KVC ...

  7. iOS KVC setValuesForKeysWithDictionary的使用

    Key Value Coding Key Value Coding是cocoa的一个标准组成部分,它能让我们可以通过name(key)的方式访问property, 不必调用明确的property ac ...

  8. iOS.KVC.setValue:forKey:

    Foundation Framework 定义了 NSObject(NSKeyValueCoding), - (void)setValue:(id)value forKey:(NSString *)k ...

  9. iOS KVC 和 KVO 区别简单总结

    KVC: key value coding,键值编码.是一种通过使用属性的名称(key)来间接访问对象属性的方法.这个方法可以不用通过 setter/getter 方法来访问对象的属性.该方法使用的实 ...

随机推荐

  1. Java算法分析1—————寻找数组同样元素

    算法的两个评測指标:执行时间和内存消耗 要么用时间换空间,要么用空间换时间 寻找数组同样元素測试一: 0~99共100个元素各不同样,新增加一个0~99的元素不明白位置 从101个元素数组中找出与0~ ...

  2. sharepoint 2013 userprofile 用户信息

    Sharepoint2013获得当前用户userfrofile 基本介绍: 什么使用户配置文件. 用户属性和用户配置文件属性提供有关 SharePoint 用户的信息,如显示名称.电子邮件.标题以及其 ...

  3. alertify、js、css 使用简介

    Alertify.js which helped me resolve my issues regarding prompts, alerts, confirms, etc in iOS7. 1.al ...

  4. 搭建ganglia集群而且监视hadoop CDH4.6

    前言 近期在研究云监控的相关工具,感觉ganglia颇有亮点,能从一个集群总体的角度来展现数据. 但是安装过程稍过复杂,相关依赖稍多,故写此文章与大家分享下. 本文不解说相关原理,若想了解请參考其它资 ...

  5. Android SDK 5.0 这个语句带来折腾 - 生命在于折腾!

    Android SDK 5.0  带来的这番折腾 - 生命在于折腾! 太阳火神的漂亮人生 (http://blog.csdn.net/opengl_es) 本文遵循"署名-非商业用途-保持一 ...

  6. in与exist , not in与not exist 的区别(转)

    in和exists  in 是把外表和内表作hash 连接,而exists是对外表作loop循环,每次loop循环再对内表进行查询.一直以来认为exists比in效率高的说法是不准确的.  如果查询的 ...

  7. 我的学习笔记_Windows_HOOK计划 2009-12-03 11:19

    一.什么是HOOK? "hook"这个单词的意思是"钩子","Windows Hook"是Windows消息处理机制的一个重要扩展,程序猿能 ...

  8. vs2015 安装问题汇总

    1. The product version that you are trying to set up is earlier than the version already installed o ...

  9. proxy pattern 代理模式

    常用的几种代理模式简要说明如下:  (1) 远程代理(Remote Proxy):为一个位于不同的地址空间的对象提供一个本地的代理对象,这个不同的地址空间可以是在同一台主机中,也可是在另一台主机中,远 ...

  10. Java Metrics

    Java Metrics Java Metrics是一个功能比較强大的java统计库,它的输出组件也非常强大,帮我们做好了: 输出到Ganglia 输出到控制台 输出到JMX 输出Json 具体见:d ...