一:KVC基础使用:(KVC主要对三种类型进行操作:基础数据类型及常量,对象类型,以及集合类型)
1):KVC setter方法:
setValue:forKey: (属性为key,赋值)
setValue:forKeyPath: (多级访问,不仅可以对当前对象属性进行赋值,也可以对“深层”对象的属性进行赋值)
2):KVC getter方法:
valueForKey:
valueForKeyPath:(多级访问,不仅可以对当前对象属性进行赋值,也可以对“深层”对象的属性进行取值)
释例如下:
Person *person = [[Person alloc] init];
[person setValue:@"张三" forKey:@"name"];
NSString *name = [person valueForKey:@"name"];
NSLog(@"person name = %@",name);
[person setValue:@"小黄" forKeyPath:@"pet.name"];
NSString *petName = [person valueForKeyPath:@"pet.name"];
NSLog(@"pet name = %@",petName);
3):当集合中元素类型一致,可以通过KVC的getter方法valueForKeyPath:来获取所有集合元素中的属性,生成一个新的元素属性集合。(这里的集合一般为NSArray,NSSet,不包括NSDictionary)
释例如下:
Person *person1 = [[Person alloc] init];
person1.name = @"李四";
person1.pet.name = @"小白";
NSArray *personArray = [NSArray arrayWithObjects:person,person1, nil];
NSArray *personNameArray = [personArray valueForKeyPath:@"name"];
NSLog(@"person name array = %@",personNameArray);
NSArray *petNameArray = [personArray valueForKeyPath:@"pet.name"];
NSLog(@"pet name array = %@",petNameArray);
4):KVC多值操作:(模型转字典)
dictionaryWithValuesForKeys:通过一组key集合,来获取以集合中的值为key,模型中的属性值为value的字典。setValuesForKeysWithDictionary:通过一组字典(模型中的属性名为key,属性值为value),对模型进行赋值
释例如下:
NSArray *keyArray = @[@"name",@"age",@"pet"];
NSDictionary *dictionary = [person dictionaryWithValuesForKeys:keyArray];
NSLog(@"dictionary = %@",dictionary);
NSDictionary *keyValuesDict = @{@"name":@"王五",@"age":@18};
[person setValuesForKeysWithDictionary:keyValuesDict];
NSLog(@"%s",__func__);
5):KVC异常:
a):当对非对象类型赋值为nil时,系统会调用setNilValueForKey:方法,抛出NSInvalidArgumentException异常,程序crash;
b):当对对象赋值,取值时,用到未定义的key时,系统会调用setValue:forUndefinedKey:和valueForUndefinedKey:方法,抛出NSUndifineKeyException异常,程序crash
二:KVC的集合操作:
1):集合属性操作
当集合中的元素类型一致时,可以通过KVC机制中的mutableArrayValueForKey:,mutableOrderedSetValueForKey:,mutableSetValueForKey:等方法,获取集合中元素属性的集合。
注意:
KVO的实现原理是:在运行时时,创建观察者对应类的派生类,并且把观察实例对象的isa指针指向新创建的派生类,在这个派生类中重写待观察属性的setter方法,在set属性值之前或之后发送属性值改变的消息。
我们一般可以通过KVO机制,来观察实例对象属性,从而进行属性值改变后的操作。但是,在对集合对象进行add,remove等元素操作时,并不会出发KVO的观察者机制。(KVO机制是通过重写setter方法,达到观察目的,它只会在对对象属性进行直接赋值时,进行触发。)
通过KVC机制,获取到集合元素的属性值集合,通过KVO观察这些属性值时,在该属性值集合中进行add,remove等元素操作时,依然可以触发KVO的观察者机制。
2):集合运算符:
KVC中的valueForKeyPath:不仅可以通过.语法深层的访问对象的属性,还可以对调用该方法的对象进行“深层”操作
a);格式:object.@operator.property
object:要执行集合运算的对象
operator:集合运算符
property:要执行运算的属性
b):集合运算符分为三类:
集合操作符:@avg,@count,@sum,@max,@min
数组操作符:@unionOfObject(不去重),@distinctUnionOfObject(去重),功能均为返回属性值集合
三:KVC的搜索机制:
1):getter搜索:
a):通过getter方法搜索,按照get<key>,<key>,is<key>,_<key>顺序搜索,若没有找到,进入第二步;
b):查找匹配方法:countOf<key>,objectOf<key>AtIndex,<key>AtIndexs:
如果找到第一个方法,以及2,3方法中任意一个,则创建一个集合对象,该对象响应所有NSArray的方法并返回对象;
如果没有找到(NSArray简单存取方法,及NSArray存取方法组),进入第三步;
c):查找有没有countOf<key>,enumeratorOf<key>,memberOf<key>命名的方法:
若找到三个方法,则创建一个集合对象,响应NSSet方法并返回;
如果取回的是一个对象指针,则直接返回;
如果取回的是基础数据类型,并支持NSNumber类型,则存储为SNSNumber并返回;
如果返回的对象不支持NSNumber类型,则存储为NSValue并返回;
若果上述情况均未找到时,则调用valueForUndefinedKey:方法并抛出异常
四:KVC的性能及私有变量访问
1):通过KVC机制的搜索原则,我们可以知道,通过KVC机制获取属性值的性能比通过getter方法获取属性值的性能要差。所以,在使用KVC机制时,我们一般不设置该属性的setter,getter方法,以提高搜索性能。
2):我们一般不用KVC机制来实现集合操作,因为集合操作所消耗的性能更大。
3):KVC的本质是操作类的方法列表,以及查找实例变量。所以,我们可以通过KVC机制,来实现私有变量的访问。(例如readonly修饰的属性,@protected修饰的属性)
4):当我们不需要私有变量能够被访问时,我们可以重写accessInstanceVariablesDirectly方法,设置为NO。(默认为YES,表明可以通过KVC访问私有变量)