参考链接:
[精辟]KVO 和 KVC 的区别和用法,干货!!!
iOS开发技巧系列---详解KVC
KVC的优点和缺点
KVC
KVC是键值编码(key-value coding),cocoa的标准组成部分, 是一种可以直接通过字符串的名字(Key)来访问类属性的机制,而不是通过直接调用setter,getter方法来访问.
KVC的具体用法
1.访问类的私有变量
无论类中的成员是否私有,用KVC都可以强行"破门而入",对他正常的取值和赋值.
比如一个Dog类
@interface Dog : NSObject
{
@private
double height;
}
可以通过setValue:forKey:方法来给私有变量赋值(这里关键字Key就是height字符串)
[dog setValue:@12 forKey:@"height"];
可以通过valueForKey:方法来读取私有变量的值.
[dog valueForKey:@"height"];
如果属性本身是一个类.
比如
@property (nonatomic,strong) Dog *pet;
我们可以通过带Path的方法来取到这个属性.
[person setValue:@12 forKeyPath:@"pet.height"];
2.使用KVC将字典(或者说json)转化为模型
NSDictionary *dic = @{
@"name":@"jack"
@"money":@"20.7"
}
//创建模型,假定模型已有两个属性name和money
Person *p = [p new];
//转换类型
[p setValuesForKeysWithDictionary:dic];
NSLog(@"%@",p.name);
3.使用KVC直接访问NSArray或者NSSet的属性值
//把三本书存入books数组.
NSArray *books = @[book1,book2,book3];
//把书名取出来存到另一个数组中
NSArray *names = [books valueForKeyPath:@"name"];
//打印加个的平均值
NSLog(@"%@",[books valueForKeyPath:@"avg.price"]);
KVO
详情请看设计模式那一部分.
KVO&KVC常见面试题
问题1:KVC的底层实现(或者,kvc的查找顺序)
当一个对象调用setValue方法时,方法内部会做以下操作:
①..检查是否存在相应key的set方法,如果存在,就调用set方法
②...如果set方法不存在,就会查找与key相同名称并且带下划线的成员属性,如果有,则直接给成员属性赋值
③....如果没有找到_key,就会查找相同名称的属性key,如果有就直接赋值
④.....如果还没找到,则调用valueForUndefinedKey:和setValue:forUndefinedKey:方法。这些方法的默认实现都是抛出异常,我们可以根据需要重写它们。
问题2:KVO的底层实现
kvo基于runtime机制实现。
使用了isa 混写(isa-swizzling),当一个对象(假设是person对象,person的类是MYPerson)的属性值(假设person的age)发生改变时,系统会自动生成一个类,继承自MYPerson :NSKVONotifying_MYPerson,在这个类的setAge方法里面,调用[super setAge:age] [self willChangeValueForKey:@"age"] 和 [self didChangeValueForKey:@"age"]
,而这两个方法内部会主动调用监听者内部的 - (void)observeValueForKeyPath 这个方法。想要看到NSKVONotifying_MYPerson很简单,在self.person.age = 20; 这里打断点,在调试区域就能看到 _person->NSObject->isa=(Class)NSKVONotifying_MYPerson.同时我们在 self.person = [[MYPerson alloc]init];后面打断点,看到_person->NSObject->isa=(Class)MYPerson,由此可见,在添加监听者之后,person类型已经由MYPerson被改变成NSKVONotifying_MYPerson
问题3:什么是KVO和KVC?
答:KVC:键 – 值编码 使用字符串直接访问对象的属性。
KVO:键值观察机制,它提供了观察某一属性变化的方法