iOS

KVC原理及使用小技巧

2018-08-15  本文已影响74人  为什么划船不靠桨

相信大多数人都已经能够很清楚的了解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访问属性顺序(机制)是怎么样的?(对于属性name而言)
     1.先去找有没有name的set,get方法,如果有,那么通过set,get方法来访问属性(赋值和取值)
     2.如果找不到set,get方法。那么会先去找有没有一个变量是:_name。如果有,那么直接给_name复制和取值
     3.如果找不到_name。那么会去找有没有一个变量:_isName,如果有,那么直接给_isName复制和取值
     4.如果找不到_isName。那么会去找有没有一个变量:name,如果有,那么直接给name复制和取值
     5.如果找不到name。那么会去找有没有一个变量:isName,如果有,那么直接给isName复制和取值
     4.如果找不到isName。就会崩溃crash
     */
[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”];
上一篇下一篇

猜你喜欢

热点阅读