相信大多数人都已经能够很清楚的了解KVC了,但是我依旧要给大家简单的解释一下,避免有不太清楚的小伙伴看不懂.
KVC,键值编码,是一套利用字符串标识符间接访问对象属性和关系的机制,通过此机制无需通过set和get方法就能对对象的属性进行设置和获取,也可以对只读的属性进行设置.定义在NSKeyValueCoding.h
文件中,是一个非正式协议.
在NSKeyValueCoding中提供了KVC通用的访问方法,分别是getter方法valueForKey:和setter方法setValue:forKey:,以及其衍生的keyPath方法,这两个方法各个类通用的。并且由KVC提供默认的实现,我们也可以自己重写对应的方法来改变实现。
//学习KVC之后,可以这样访问属性.核心是通过可以来进行赋值和取值
[people setValue:@"李四" forKey:@"name"];
NSLog(@"name = %@",[people valueForKey:@"name"]);
/*
KVC访问setter方法的属性顺序(机制)是怎么样的?(对于属性name而言)
1.先去找有没有name的setName方法,如果有,那么通过set方法来设置属性
2.如果找不到set,get方法。假如类方法accessInstanceVariablesDirectly返回YES(NSKeyValueCodingCatogery中实现的类方法,默认实现为返回YES),那么会先去找有没有一个变量是:_name。如果有,那么直接给_name复制和取值
3.如果找不到_name。那么会去找有没有一个变量:_isName,如果有,那么直接给_isName复制和取值
4.如果找不到_isName。那么会去找有没有一个变量:name,如果有,那么直接给name复制和取值
5.如果找不到name。那么会去找有没有一个变量:isName,如果有,那么直接给isName复制和取值
4.如果找不到isName。就会崩溃crash
*/
/*
KVC访问getter方法的属性顺序(机制)是怎么样的?(对于属性name而言)
1、首先按getName,name,isName的顺序查找getter方法,找到直接调用。如果是BOOL、int等内建值类型,会做NSNumber的转换。
2、上面的getter没找到,查找countOfKey、objectInKeyAtindex、KeyAtindexes格式的方法。如果countOfKey和另外两个方法中的一个找到,那么就会返回一个可以响应NSArray所有方法的代理集合的NSArray消息方法。
3、还没找到,查找countOfKey、enumeratorOfKey、memberOfKey格式的方法。如果这三个方法都找到,那么就返回一个可以响应NSSet所有方法的代理集合。
4、还是没找到,如果类方法accessInstanceVariablesDirectly返回YES。那么按 _key,_isKey,key,iskey的顺序搜索成员名。
5、再没找到,调用valueForUndefinedKey。如果这个方法没有重写,则直接崩溃
*/
[p1 setValue:@"尼桑" forKeyPath:@"_car.carName"];
NSLog(@"carName = %@",[p1 valueForKeyPath:@"car.carName"]);
//当通过KVC给某个对象的属性赋值为nil时,此时KVC会调用属性所属对象的setNilValueForKey:方法,并抛出NSInvalidArgumentException的异常,并使应用程序Crash。
//我们可以通过重写下面方法,在发生这种异常时进行处理。例如给name赋值为nil的时候,就可以重写setNilValueForKey:方法并表示name是空的。
- (void)setNilValueForKey:(NSString *)key {
if ([key isEqualToString:@"name"]) {
[self setValue:@"" forKey:@”age”];
} else {
[super setNilValueForKey:key];
}
}
//当调用valueForKey获得key所对应的value时,没有搜索到对应的key或者keyPath,会抛出一个异常,重写下列方法即可
- (nullable id)valueForUndefinedKey:(NSString *)key;
//当调用setValue设置value的值时,没有搜索到对应的key或者keyPath,会抛出一个异常,重写下列方法即可
- (void)setValue:(nullable id)value forUndefinedKey:(NSString *)key;
集合属性操作
根据KVO的实现原理,是在运行时生成新的子类并重写其setter方法,在其内容发生改变时发送消息。但这只是对属性直接进行赋值会触发,如果属性是容器对象,对容器对象进行add或remove操作,则不会调用KVO的方法。可以通过KVC对应的API来配合使用,使容器对象内部发生改变时也能触发KVO。
在进行容器对象操作时,先调用下面方法通过key或者keyPath获取集合对象,然后再对容器对象进行add或remove等操作时,就会触发KVO的消息通知了。
- (NSMutableArray *)mutableArrayValueForKey:(NSString *)key;
- (NSMutableOrderedSet *)mutableOrderedSetValueForKey:(NSString *)key API_AVAILABLE(macos(10.7), ios(5.0), watchos(2.0), tvos(9.0));
- (NSMutableSet *)mutableSetValueForKey:(NSString *)key;
keyPath方法:
- (NSMutableArray *)mutableArrayValueForKeyPath:(NSString *)keyPath;
- (NSMutableOrderedSet *)mutableOrderedSetValueForKeyPath:(NSString *)keyPath API_AVAILABLE(macos(10.7), ios(5.0), watchos(2.0), tvos(9.0));
- (NSMutableSet *)mutableSetValueForKeyPath:(NSString *)keyPath;
很实用的KVC干货小技巧
获取数组里的,最大、最小、平均、求和
NSArray *array = @[@"1",@"3",@2,@9.5,@"1.2"];
NSNumber *sum = [array valueForKeyPath:@"@sum.floatValue"];
NSNumber *avg = [array valueForKeyPath:@"@avg.floatValue"];
NSNumber *max = [array valueForKeyPath:@"@max.floatValue"];
NSNumber *min = [array valueForKeyPath:@"@min.floatValue"];
NSLog(@"sum:%@",sum);
NSLog(@"avg:%@",avg);
NSLog(@"max:%@",max);
NSLog(@"min:%@",min);
删除重复数据
NSArray *array = @[@"name", @"w", @"aa", @"zxp", @"aa"]; //返回的是一个新的数组
NSArray *newArray = [array valueForKeyPath:@"@distinctUnionOfObjects.self"];
NSLog(@"%@", newArray);
同样可以嵌套使用,先剔除name对应值的重复数据再取值
NSArray *array = @[ @{@"title":@"zxp",@"name":@"zhangxiaoping"}, @{@"title":@"zxp2",@"name":@"zhangxiaoping2"}, @{@"title":@"zxp",@"name":@"zhangxiaoping3"}, @{@"title":@"zxp",@"name":@"zhangxiaoping"}];
//根据name字段,来进行重复删除。
NSArray *newArray = [array valueForKeyPath:@"@distinctUnionOfObjects.name"];
//如果要根据title字段来删除重名的写法为`@distinctUnionOfObjects.title`
NSLog(@"%@", newArray);
/*
print:( zhangxiaoping3, zhangxiaoping2, zhangxiaoping)是一个字符串数组*/
进行实例方法的调用,例如将数组中的所有字符串都变成大写,或者获取到每个字符串的长度
NSArray *array = @[@"name", @"w", @"aa", @"ZXPing"];
NSLog(@"%@", [array valueForKeyPath:@"uppercaseString"]);
当然,字符串的其他方法都是可以进行类推的
NSArray *array = @[@"name", @"w", @"aa", @"ZXPing"];
NSLog(@"%@", [array valueForKeyPath:@"length"]);
改变 UITextfield 的 placeholder 的颜色
[addressTextField setValue:[UIColor redColor] forKeyPath:@”_placeholderLabel.textColor”];