前言
简单来说KVC 就是通过如下方式进行间接设置和获取值的
People *p = [[People alloc]init];
[p setValue:@"xiaobing" forKey:@"name"];
[p setValue:<#(nullable id)#> forKeyPath:<#(nonnull NSString *)#>];
[p valueForKey:<#(nonnull NSString *)#>]
[p valueForKeyPath:<#(nonnull NSString *)#>];
简单使用
[p setValue:@"xiaobing" forKey:@"name"];
[p valueForKey:<#(nonnull NSString *)#>];
假设有一People 类 并且有一个属性name 这里普及一个概念,OC中属性 是@property 描述的 成员变量 是非 property描述的变量 实例变量 是一种特殊的成员变量 看如下代码
People *p = [[People alloc]init];
[p setValue:@"xiaobing" forKey:@"name"];
[p valueForKey:@"name"]; // 结果就是 xiaobing
假设又个Son类 继承People
Son *son = [[Son alloc]init];
[p setValue:son forKey:@"s"];
[p setValue:@"sonName" forKeyPath:@"s.name"];
NSLog(@"---s.name---%@",[p valueForKeyPath:@"s.name"]); // sonName
设置值时属性查找顺序
还以name为例 在setValue forKey中 首先看有没有setName _setName _setIsName 方法 如果有就直接调用 比如我们的属性 就会直接调用set方法 如果没有 再检查accessInstanceVariablesDirectly方法 (默认返回YES 如果返回NO则不允许间接访问 那么就不会走下边的流程)就会按照 _name _isName name isName 的成员变量 然后赋值 简单举例
** 这里我们将属性name 变成_name 的成员变量 **
通过调用
[p setValue:@"xiaobing" forKey:@"name"];
NSLog(@"---p.name--%@",[p valueForKey:@"name"]);
仍然可以拿到正确的值 其他几种情况自行验证。。
接下来一步一步验证 首先假设People中没有成员变量school
通过在People类中添加如下三个方法
- (void)setSchool:(NSString *)school{
NSLog(@"--setSchool---%s",__func__);
}
- (void)_setSchool:(NSString *)school{
NSLog(@"--_setSchool---%s",__func__);
}
- (void)setIsSchool:(NSString *)school{
NSLog(@"--isSchool---%s",__func__);
}
分别注释掉其中的两个方法 程序会调用其中的一个 不过此时程序会崩溃 因为设置了不存在的key 关于异常处理 后边再讲。 接下来会按照_name _isName name isName的流程查找赋值 接下来验证
People 类中添加 成员变量
@public
NSString *_school;
NSString *_isSchool;
NSString *school;
NSString *isSchool;
执行
[p setValue:@"浙江大学" forKey:@"school"];
NSLog(@"----1----%@%@%@%@",p->_school,p->_isSchool,p->school,p->isSchool);//----1----浙江大学(null)(null)(null)
** 此时要将set方法注释掉 不然不会执行到这里 **
获取值查找顺序
首先会找get方法 比如 getName name isName 的get方法 然后 继续会找countOfName objectInNameAtIndex 判断 accessInstanceVariablesDirectly 是否返回为true 最后按照 _name _isName name isName 的成员变量 然后获取到指定的值
接下来一一验证 还以school为例 在people类的.m中添加如下三个方法
- (NSString *)getSchool{
return @"-----getSchool school";
}
- (NSString *)school{
return @"----school school";
}
- (NSString *)isSchool{
return @"----isSchool school";
}
注释掉其中任意两个 调用 NSLog(@"—school—%@",[p valueForKeyPath:@“school”]);都能返回相应的结果 -----getSchool school 或者 ----school school 或者 ----isSchool school
再来验证下 countOfName objectInNameAtIndex enumeratorOfClasses
在people类中添加一个属性students
@property(nonatomic,retain)NSMutableArray *students;
将students赋上值 p.students = [[NSMutableArray alloc]initWithObjects:@“s1”,@“s2”,@“s3”, nil];
这个时候我们使用
NSArray *arr = [p valueForKey:@"classes"];
** 注意此时我们获取的是 classes ** 依次创建下边三个方法
1.
- (NSInteger)countOfClasses{
NSLog(@"-----countOfClasses---");
return self.students.count;
}
2.
- (id)objectInClassesAtIndex:(NSUInteger)index{
NSLog(@"-----objectInClassesAtIndex---");
return self.students[index];
}
3.
// 迭代器
- (id)enumeratorOfClasses {
// objectEnumerator
return [self.students reverseObjectEnumerator];
}
此时会一次调用 countOfClasses 我们返回的是 students的数量 objectInClassesAtIndex 返回的是 students 中的成员。对于第三个迭代器是在便利时用的
NSEnumerator *enumerator = [arr objectEnumerator];
NSString *str = nil;
while (str = [enumerator nextObject]) {
NSLog(@"---3-----%@",str);
}
KVC 异常处理
- 设置了不存在的key 造成崩溃 只需要实现如下方法
- (void)setValue:(id)value forUndefinedKey:(NSString *)key{
NSLog(@"-----设置了不存在的key-----%@",key);
}
- 设置了nil 造成崩溃 只需要实现如下方法
- (void)setNilValueForKey:(NSString *)key{
NSLog(@"-----设置了nil");
}
- 获取了不存在的key 只需要实现如下方法
- (id)valueForUndefinedKey:(NSString *)key{
NSLog(@"----获取了不存在的key---%@",key);
return nil;
}
- 设置的值的类型不匹配 造成崩溃 这种情况下 只能自己判断好类型
KVC高级用法
- setValuesForKeysWithDictionary
people.h
@property(nonatomic,copy)NSString *name;
@property(nonatomic,assign)NSInteger age;
使用
[p setValuesForKeysWithDictionary:@{@"name":@"xiaobing哈",@"age":@18}];
NSArray *keys = @[@"name",@"age"];
NSLog(@"------%@",[p dictionaryWithValuesForKeys:keys]);
// 结果
{
age = 18;
name = "xiaobing\U54c8";
}
还有很多 这里就介绍这一种 也是最近项目中用到的 需要把一个模型的数据 同步专成另一个模型 用这个很方便。