这两天晚上在学习腾讯公开课的视频,其中在1月18号的内容是KVC的底层实现。看完后简单记一下学习笔记。
1、[obj valueForKey:key]的搜索路径
第一步,首先会搜索getter方法(3个方法)。优先级 -getKey > Key > isKey。
第二步,若找不到找getter 方法 ,则寻找+(BOOL)accessInstanceVariablesDirectly;方法。如果类方法返回NO,跳过第三步。
第三步,寻找相关的成员变量(4个)。优先级_key > _isKey > key > isKey。
第四步,还没找到的话,查找countOfKey、enumeratorOfKey、memberOfKey格式的方法。如果这三个方法都找到,那么就返回一个可以响应NSSet所有方法的代理集合。
如果以上方法都拿不到,调用valueForUndefinedKey。
2、[obj setValueForKey:key]的搜索路径
第一步,搜索setter相关方法(2个方法)。优先级setKey > setIsKey。
第二步,寻找+(BOOL)accessInstanceVariablesDirectly;方法如果返回NO,跳过第三步
第三步,寻找相关的成员变量(4个)。优先级_key > _isKey > key > isKey。
如果没有找到成员变量,调用setValue:forUnderfinedKey:
3、KVC防崩溃处理
分别实现 valueForUndefinedKey 和 setValue:forUnderfinedKey: 以及setNilValueForKey方法。
4、通过KeyPath来设值
在开发过程中,一个类的成员变量有可能是其他的自定义类,你可以先用KVC获取出来再该属性,然后再次用KVC来获取这个自定义类的属性,但这样是比较繁琐的,对此,KVC提供了一个解决方案,那就是键路径KeyPath
- (void)setValue:(nullable id)value forKeyPath:(NSString *)keyPath;
//第一步就是分离key,用小数点.来分割key,然后再像普通key一样按照先前介绍的顺序搜索下去。
5、KVC容器方法
- (NSMutableArray*)mutableArrayValueForKey:(NSString*)key;
//这是集合操作的API,里面还有一系列这样的API,如果属性是一个NSMutableArray,那么可以用这个方法来返回。
6、集合运算符
简单集合运算符:@avg、@sum、@max、@min、@count (只能用在集合对象中,对象属性必须为数字类型)
NSInteger count = [[perArray valueForKeyPath:@"@count.age"] integerValue];
7、对象操作符
@unionOfObjects:返回指定属性的值的数组,不去重
NSArray *nameArray = [perArray valueForKeyPath:@"@unionOfObjects.name"];
@distinctUnionOfObjects:返回指定属性去重后的值的数组
NSArray *nameArray2 = [perArray valueForKeyPath:@"@distinctUnionOfObjects.name"];
8、数组 / 集体操作符
跟对象操作符很相似,只不过是在NSArray和NSSet所组成的集合中工作的。
// 数组 / 集体操作符 : 对由NSArray和NSSet所组成的集合操作
// unionOfArrays:返回一个数组,值由各个子数组的元素组成,不去重
// distinctUnionOfArrays:返回一个数组,值由各个子数组的元素组成,去重
// distinctUnionOfSets:和distinctUnionOfArrays差不多, 只是它期望的是一个包含着NSSet对象的NSSet,并且会返回一个NSSet对象。因为集合不能包含重复的值,所以它只有distinct操作。
Person *P1 = [[Person alloc] initWithName:@"PP1"andAge:10];
Person *P2 = [[Person alloc] initWithName:@"PP1"andAge:20];
Person *P3 = [[Person alloc] initWithName:@"PP2"andAge:30];
Person *P4 = [[Person alloc] initWithName:@"PP3"andAge:40];
Person *P5 = [[Person alloc] initWithName:@"PP3"andAge:50];
NSArray *perArray2 = @[P1, P2, P3, P4, P5];
NSArray *arr1 = [@[perArray,perArray2] valueForKeyPath:@"@unionOfArrays.name"];
NSArray *arr2 = [@[perArray,perArray2] valueForKeyPath:@"@distinctUnionOfArrays.name"];