iOS - KVC 键值编码
1、KVC
KVC 是 Key-Value Coding 的简写,是键值编码的意思,属于 runtime 方法。Key Value Coding 是 cocoa 的一个标准组成部分,是间接给对象属性设置数值的方法,它能让我们可以通过 name(key) 的方式访问属性变量, 不必调用明确的属性访问方法, 如我们有个属性变量叫做 foo, 我们可以 foo 直接访问它,同样我们也可以用 KVC 来完成 [Object valueForKey:@“foo”], 这样做主要的好处就是来减少我们的代码量。
程序执行过程中,KVC 动态给对象属性设置数值,不关心属性在 .h 和 .m 中是如何定义的,只要对象有属性,就能够读取和设置。这种方式,有点违背程序的开发原则。
在 iOS 中,用 KVC 用的最多是核心动画,核心动画是通过 KVC 对涂层的可动画属性设置数值来实现的。
2、数据模型
模型是专门用来存放数据的对象,一般都是一些直接继承自 NSObject 的纯对象,内部会提供一些属性来存放数据,控制器可以直接传递模型给视图控件以显示空间的内容。
1)用模型取代字典的好处
使用字典的坏处:
一般情况下,设置数据和取出数据都使用 “字符串类型的 key”,编写这些 key 时,编辑器没有智能提示,需要手敲,手敲字符串 key,key 容易写错,Key 如果写错了,编译器不会有任何警告和报错,造成设错数据或者取错数据。如:
dict[@"name"] = @"Jack";
NSString *name = dict[@"name"];
使用模型的好处:
所谓模型,其实就是数据模型,专门用来存放数据的对象,用它来表示数据会更加专业。模型设置数据和取出数据都是通过它的属性,属性名如果写错了,编译器会马上报错,因此,保证了数据的正确性
使用模型访问属性时,编译器会提供一系列的提示,提高编码效率。app.name = @"Jack";
NSString *name = app.name;
2)字典转模型
字典转模型的过程最好封装在模型内部。
模型应该提供一个可以传入字典参数的构造方法。
- (instancetype)initWithDict:(NSDictionary *)dict;
+ (instancetype)xxxWithDict:(NSDictionary *)dict;
3)字典转模型 KVC 方法
- 字典转模型:setValuesForKeysWithDictionary
- 字典中的 key 值需与要赋值的对象的属性变量名相同,并且都为字符串类型。
- 模型转字典:dictionaryWithValuesForKeys
- 参数是要被转换到字典中的属性名称
- 字典转模型:setValuesForKeysWithDictionary
4)KVC 数据模型的设置
为了避免服务端返回的数值型数据是 null,可以把数值型的数据设置成 NSNumber 类型,否则会报错 could not set nil as the value for the key messageId 。
id 是服务端最喜欢用的属性,id 在 iOS 中是关键字,但在模型中可以正常使用的。
copy 属性,在设置数值的时候,如果有一方是可变的,会默认做一次 copy 操作,会建立新的副本,在模型中对象全都是用 copy 属性会比较安全。
定义为 copy 的属性,重写了 setter 方法之后,定义属性的 copy 就是摆设了,不会默认进行 copy 操作,必须要自己 copy 一下,否则设置数值的时候,不会 copy。
5)字典转模型的过程
3、KVC 赋值与取值
KVC 是一种操作全局变量的方法,无论是公有的,私有的,还是受保护的全都可以操作。
- 1、找对象的 setter 方法,找到就执行;
- 2、找不到 setter 方法就找 _name 变量,找到就赋值;
- 3、如果找不到 _name 变量,就找 name;
- 4、如果 name 也找不到就会让对象调用 -(void)setValue:forUnderfinedKey; 方法处理异常。
Objective-C
KvcClass.h
@property(nonatomic, assign) NSInteger ID;
@property(nonatomic, copy) NSString *name;
@property(nonatomic, assign) NSInteger age;
@property(nonatomic, retain) SubKvcClass *subKVC;
SubKvcClass.h
@property(nonatomic, copy) NSString *subName;
@property(nonatomic, assign) NSInteger subAge;
ViewController.m
KvcClass *kvcObject = [[KvcClass alloc] init];
SubKvcClass *subKVCObject = [[SubKvcClass alloc] init];
kvcObject.subKVC = subKVCObject;
Swift
KvcClass.swift
var ID:NSInteger!
var name:String!
var age:NSInteger = 0
var subKVC:SubKvcClass!
SubKvcClass.swift
var subName:String!
var subAge:NSInteger = 0
ViewController.swift
var kvcObject = KvcClass()
var subKVCObject = SubKvcClass()
kvcObject.subKVC = subKVCObject
3.1 通过 键值编码 给对象的属性动态赋值
必须得有标准的 getter 和 setter 方法,或者用 @property 声明。
调用 setValue: forKey: 方法以字符串的方式向对象发送消息,设置实例变量的值。第一个参数是要设置的值,第二个参数是实例变量的名称。
调用 valueForKey: 来获取实例变量的值。
Objective-C
// 动态设置属性的值
[kvcObject setValue:@"xiao bai" forKey:@"name"];
[kvcObject setValue:@"8" forKey:@"age"];
// 获取实例变量的值
NSString *name = [kvcObject valueForKey:@"name"];
NSInteger age = [[kvcObject valueForKey:@"age"] integerValue];
Swift
// 动态设置属性的值
kvcObject.setValue("xiao bai", forKey: "name")
kvcObject.setValue("8", forKey: "age")
// 获取实例变量的值
let name1 = kvcObject.valueForKey("name") as! String
let age1 = kvcObject.valueForKey("age") as! NSInteger
3.2 通过 键路径 给实例变量是其他类的对象赋值
如果实例变量中有其他类的对象,那么可以使用 setValue: forKeyPath: 给其他类的对象的属性变量赋值。
Objective-C
// 通过键路径给 KVCClass 中的对象的属性赋值
[kvcObject setValue:@"sub xiao bai" forKeyPath:@"subKVC.subName"];
[kvcObject setValue:@"5" forKeyPath:@"subKVC.subAge"];
// 获取 KVCClass 中的对象的属性值
NSString *subName = [kvcObject valueForKeyPath:@"subKVC.subName"];
NSInteger subAage = [[kvcObject valueForKeyPath:@"subKVC.subAge"] integerValue];
Swift
// 通过键路径给 KvcClass 中的对象的属性赋值
kvcObject.setValue("sub xiao bai", forKeyPath: "subKVC.subName")
kvcObject.setValue("5", forKeyPath: "subKVC.subAge")
// 获取 KvcClass 中的对象的属性值
let subName1 = kvcObject.valueForKeyPath("subKVC.subName") as! String
let subAage1 = kvcObject.valueForKeyPath("subKVC.subAge") as! NSInteger
3.3 通过 字典 给对象的属性动态赋值
字典中的 key 值需与要赋值的对象的属性变量名相同。并且都为字符串类型。
- (void)setValuesForKeysWithDictionary:(NSDictionary<NSString *, id> *)keyedValues;
- (NSDictionary<NSString *, id> *)dictionaryWithValuesForKeys:(NSArray<NSString *> *)keys;
Objective-C
NSDictionary *kvcDic = @{@"name":@"xiaobai", @"age":@"6"};
// 以字典的 key 和 value 值分别作为 kvc 的 key 和 value 存储
[kvcObject setValuesForKeysWithDictionary:kvcDic];
// 取值,获取指定 keys 值对应的 values
NSDictionary *dicValue = [kvcObject dictionaryWithValuesForKeys:@[@"name", @"age"]];
Swift
let kvcDic = ["name":"xiaobai", "age":"6"]
// 以字典的 key 和 value 值分别作为 kvc 的 key 和 value 存储
kvcObject.setValuesForKeysWithDictionary(kvcDic)
// 取值,获取指定 keys 值对应的 values
let dicValue = kvcObject.dictionaryWithValuesForKeys(["name", "age"])
4、KVC 异常处理
4.1 数据冗余处理
在键值编码的类中使用以下两个方法处理 key 值不存在的异常。如果不做处理,编译时系统会报错。
- (void)setValue:(id)value forUndefinedKey:(NSString *)key;
- (id)valueForUndefinedKey:(NSString *)key;
Objective-C
KvcClass.m
// 设置不存在 key 的值
- (void)setValue:(id)value forUndefinedKey:(NSString *)key {
NSLog(@"key 值 %@ 不存在,无法设置值 !", key);
}
// 获取不存在的 key 的值
- (id)valueForUndefinedKey:(NSString *)key {
NSLog(@"key 值 %@ 不存在,无法获取值 !", key);
return nil;
}
ViewController.m
// 对象 kvcObject 没有 score 属性,出现数据异常
[kvcObject setValue:@"99" forKey:@"score"];
[kvcObject valueForKey:@"score"];
NSDictionary *kvcDic1 = @{@"name":@"xiaobai", @"age":@"6", @"score":@"100"};
// 对象 kvcObject 没有 score 属性,出现数据异常
[kvcObject setValuesForKeysWithDictionary:kvcDic1];
Swift
KvcClass.swift
// 设置不存在 key 的值
override func setValue(value: AnyObject?, forUndefinedKey key: String) {
print("key 值 \(key) 不存在,无法设置值 !")
}
// 获取不存在的 key 的值
override func valueForUndefinedKey(key: String) -> AnyObject? {
print("key 值 \(key) 不存在,无法获取值 !")
return nil
}
ViewController.swift
// 对象 kvcObject 没有 score 属性,出现数据异常
kvcObject.setValue("99", forKey: "score")
kvcObject .valueForKey("score")
// 对象 kvcObject 没有 score 属性,出现数据异常
let kvcDic1 = ["name":"xiaobai", "age":"6", "score":"99"]
kvcObject.setValuesForKeysWithDictionary(kvcDic1)
4.2 key 为系统关键字处理
在需要处理的数据源中有系统关键字时,在键值编码处理的类中使用以下两个方法处理 key 值为系统关键字的情况。
- (void)setValue:(id)value forUndefinedKey:(NSString *)key;
- (id)valueForUndefinedKey:(NSString *)key;
id 是服务端最喜欢用的属性,id 在 iOS 中是关键字,但在模型中可以正常使用的。
Objective-C
KvcClass.m
// 设置不存在 key 的值
- (void)setValue:(id)value forUndefinedKey:(NSString *)key {
// id 为系统关键字,用 ID 代替与系统冲突的 id
if ([key isEqualToString:@"id"]) {
// 设置自定义的 key 的值
self.ID = [(NSString *)value integerValue];
}
}
// 获取不存在的 key 的值
- (id)valueForUndefinedKey:(NSString *)key {
// id 为系统关键字,用 ID 代替与系统冲突的 id
if ([key isEqualToString:@"id"]) {
// 获取自定义的 key 的值
return [NSString stringWithFormat:@"%li", self.ID];
}
return nil;
}
ViewController.m
// id 为系统关键字
[kvcObject setValue:@"3" forKey:@"id"];
NSInteger ID1 = [[kvcObject valueForKey:@"id"] integerValue];
NSDictionary *kvcDic2 = @{@"name":@"xiaobai", @"age":@"6", @"id":@"5"};
[kvcObject setValuesForKeysWithDictionary:kvcDic2];
NSInteger ID2 = kvcObject.ID;
Swift
KvcClass.swift
// 设置不存在 key 的值
override func setValue(value: AnyObject?, forUndefinedKey key: String) {
// id 为系统关键字,用 ID 代替与系统冲突的 id
if key == "id" {
// 设置自定义的 key 的值
self.ID = (value as! NSString).integerValue
}
}
// 获取不存在的 key 的值
override func valueForUndefinedKey(key: String) -> AnyObject? {
// id 为系统关键字,用 ID 代替与系统冲突的 id
if key == "id" {
// 获取自定义的 key 的值
return NSString(format: "%li", self.ID)
}
return nil
}
ViewController.swift
// id 为系统关键字
kvcObject.setValue("3", forKey: "id")
let ID1 = (kvcObject.valueForKey("id") as! NSString).integerValue
// id 为系统关键字
let kvcDic2 = ["name":"xiaobai", "age":"6", "id":"5"]
kvcObject.setValuesForKeysWithDictionary(kvcDic2)
let ID2 = kvcObject.ID
5、KVC 消息传递
valueForKey: 的使用并不仅仅用来取值那么简单,还有很多特殊的用法,集合类也覆盖了这个方法,通过调用 valueForKey: 给容器中每一个对象发送操作消息,并且结果会被保存在一个新的容器中返回,这样我们能很方便地利用一个容器对象创建另一个容器对象。另外,valueForKeyPath: 还能实现多个消息的传递。
Objective-C
NSArray *array = @[@"10.11", @"20.22"];
// 结果为 (10, 20)
NSArray *resultArray = [array valueForKeyPath:@"doubleValue.intValue"];
Swift
let array:NSArray = ["10.11", "20.22"]
// 结果为 (10, 20)
let resultArray:AnyObject? = array.valueForKeyPath("doubleValue.intValue")
6、KVC 字典转模型 数据冗余处理
- 字典中元素与模型中的属性数量不想等的情况处理。处理 key 值不存在的异常。如果不做处理,编译时系统会报错。
6.1 系统方式
Objective-C
+ (instancetype)newsModelWithDict:(NSDictionary *)dict {
id obj = [[self alloc] init];
[obj setValuesForKeysWithDictionary:dict];
return obj;
}
/// 重写系统方法
- (void)setValue:(id)value forUndefinedKey:(NSString *)key {
}
6.2 列举属性数组方式
Objective-C
+ (instancetype)newsModelWithDict:(NSDictionary *)dict {
id obj = [[self alloc] init];
[obj setValueWithDict:dict];
return obj;
}
- (instancetype)setValueWithDict:(NSDictionary *)dict {
// 列出所有使用的属性
NSArray *properties = @[@"title", @"digest", @"imgsrc", @"replyCount"];
for (NSString *key in properties) {
// 判断字典中是否包含 key
if (dict[key] != nil) {
// 每一个属性使用 kvc 设置数值
[self setValue:dict[key] forKey:key];
}
}
return self;
}
6.3 运行时动态获取对象属性方式
Objective-C
+ (instancetype)newsModelWithDict:(NSDictionary *)dict {
id obj = [[self alloc] init];
[obj setValueWithDict:dict];
return obj;
}
/// 使用运行时动态获取对象属性
- (instancetype)setValueWithDict:(NSDictionary *)dict {
unsigned int count = 0;
// 拷贝对象属性数组(数组名就是指向数组第一个元素的地址)
objc_property_t *properties = class_copyPropertyList(self.class, &count);
// 遍历数组
for (unsigned int i = 0; i < count; ++i) {
// 从数组中获取属性
objc_property_t pty = properties[i];
// 获取属性名称
const char *cname = property_getName(pty);
NSString *key = [NSString stringWithUTF8String:cname];
if (dict[key] != nil) {
[self setValue:dict[key] forKey:key];
}
}
// 释放数组
free(properties);
return self;
}
iOS - KVC 键值编码的更多相关文章
- ios中键值编码kvc和键值监听kvo的特性及详解
总结: kvc键值编码 1.就是在oc中可以对属性进行动态读写(以往都是自己赋值属性) 2. 如果方法属性的关键字和需要数据中的关键字相同的话 ...
- Objective-C(十七、KVC键值编码及实例说明)——iOS开发基础
结合之前的学习笔记以及參考<Objective-C编程全解(第三版)>,对Objective-C知识点进行梳理总结.知识点一直在变,仅仅是作为參考,以苹果官方文档为准~ 十七.键值编码 K ...
- 使用KVC键值编码
掌握KVC从不使用setter.getter.点语法开始. ----- 前言 对象的内部状态是由属性进行封装的.访问对象属性的方式平时在开发过程中用得最多的是getter方法和点语法.键值编程KVC也 ...
- KVC - 键值编码
[基本概念] 1.键值编码是一个用于间接访问对象属性的机制,使用该机制不需要调用存取方法和变量实例就可访问对象属性. 2.键值编码方法在OC非正式协议(类目)NSKeyValueCoding中被声明, ...
- Property属性, KVC键值编码OC…
1.属性:帮你自动生成setter 和 getter 方法 属性的声明:(写在.h中) 格式: @property 数据类型 属性名 属性的实现:(写在.m中) ...
- 09 (OC)* 键路径(keyPath)、键值编码(KVC)、键值观察(KVO)
键路径在一个给定的实体中,同一个属性的所有值具有相同的数据类型.键-值编码技术用于进行这样的查找—它是一种间接访问对象属性的机制. - 键路径是一个由用点作分隔符的键组成的字符串,用于指定一个连接在一 ...
- 路径(keyPath)、键值编码(KVC)和键值观察(KVO)
键路径 在一个给定的实体中,同一个属性的所有值具有相同的数据类型. 键-值编码技术用于进行这样的查找—它是一种间接访问对象属性的机制. - 键路径是一个由用点作分隔符的键组成的字符串,用于指定一个连接 ...
- 《苹果开发之Cocoa编程》键-值编码和键-值观察
一.KVC 键-值编码(Key - Value Coding, KVC)是通过变量名的读取和设置变量值的一种方法,将字符串的变量名作为key来引用.NSObject定义了两个方法(KVC方法)用于变量 ...
- iOS监听模式系列之键值编码KVC、键值监听KVO的简单介绍和应用
键值编码KVC 我们知道在C#中可以通过反射读写一个对象的属性,有时候这种方式特别方便,因为你可以利用字符串的方式去动态控制一个对象.其实由于ObjC的语言特性,你根部不必进行任何操作就可以进行属性的 ...
随机推荐
- jenkins+jmeter+ant搭建接口测试平台
接口测试的重点是检查数据的交换,传递和控制管理过程以及系统间的相互逻辑依赖关系. 接口测试的流程 项目启动后,测试人员要尽早拿到接口测试文档. 开始编写接口测试用例 将接口测试用例部署到持续集成的测试 ...
- jquery plugins
jQuery官网插件 jQuery自定义滚动条样式插件 jQuery custom content scroller examples Twitter typeahead typeahead.js t ...
- AJAX 数据库实例
AJAX 用于创建动态性更强的应用程序. AJAX ASP 实例 下面的例子将演示当用户在输入框中键入字符时,网页如何与服务器进行通信: 实例 请在下面的输入框中键入字母(A - Z): 姓名: 建议 ...
- Shell之while循环
While循环的格式: while expression do command command ... done 1.计数器控制的while循环:主要用于已经准确知道要输入的数据和字符串的数目. 例子 ...
- Cacti Install
一.Cacti简介 Cacti是通过snmpget来获取数据,使用RRDtool绘画图形,而且你完全可以不需要了解RRDtool复杂的参数.它提供了非常强大的数据和用户管理功能,可以指定每一个用户能查 ...
- 记录整合sprinmvc+log4j的的过程
简介 由于进一步的学习以及便于自己更好的调试程序中遇到的错误,开始了将log4j整合到web项目中,项目是基于springmvc的,所以就做了一个springmvc和web项目的整合demo,本篇博客 ...
- 什么情况下用+运算符进行字符串连接比调用StringBuffer/StringBuilder对象的append性能好
如果在编写代码的过程中大量使用+进行字符串评价还是会对性能造成比较大的影响,但是使用的个数在1000以下还是可以接受的,大于10000的话,执行时间将可能超过1s,会对性能产生较大影响.如果有大量需要 ...
- [HTML]POST方法和GET方法
GET方法: function btn_get_click(){ var httpRequest = new XMLHttpRequest(); httpRequest.onreadystatecha ...
- [xcode]Xcode查找函数(方法)调用及被调用
参考资料:http://stackoverflow.com/questions/7145045/find-method-references-in-xcode 这个功能有的说是 Find Caller ...
- POJ 2001:Shortest Prefixes
Shortest Prefixes Time Limit: 1000MS Memory Limit: 30000K Total Submissions: 16782 Accepted: 728 ...