iOS面试题与核心基础之KVC&KVO

KVO

KVO是Key-value observing,俗称键值监听(键值观察),可以用于监听某个属性值的改变。
当我们在调用 addObserver:forKeyPath时,系统会在运行时动态创建NSKVONotifying_A类,同时将原来的类指针指向新创建的类。新创建的NSKVONotifying_A类是原来类的子类,目的是重写setter方法,新的setter加入了通知所有观察对象的代码。

当修改instance对象的属性时,会调用Foundation的_NSSetXXXValueAndNotify函数

willChangeValueForKey:
父类原来的setter
didChangeValueForKey:
内部会触发监听器(Oberser)的监听方法( observeValueForKeyPath:ofObject:change:context:

那为什么只调用didChangeValueForKey:无法成功调用Observer的监听方法呢?

原因:为了在回调中能够提供监听的属性的旧值,在willChangeValueForKey:中,会通过KVC取得旧值。没有取得旧值,调用Observer的监听方法的条件不成立。

KVC

KVC的全称是Key-Value Coding,俗称“键值编码”,可以通过一个key来访问某个属性

常见的API有

  • (void)setValue:(id)value forKeyPath:(NSString *)keyPath;
  • (void)setValue:(id)value forKey:(NSString *)key;
  • (id)valueForKeyPath:(NSString *)keyPath;
  • (id)valueForKey:(NSString *)key;
setValue:forKey 底层原理

当调用setValue:forKey时,会先顺序查找该类对象的setKey和_setKey方法,找到了就调用方方法传递参数。没找到则查看accessInstanceVariablesDirectly方法的返回值,返回值为NO则调用 setValue:forUndefinedKey:
并抛出异常 NSUnknownKeyException; 返回值为真则,顺序查找成员变量_key,_isKey,key,isKey,找到了直接赋值,没找到的话也是调用 setValue:forUndefinedKey: 并抛出异常NSUnknownKeyException`;

valueForKey: 底层原理

当调用valueForKey:时,会先顺序查找该类对象的getKey、key、isKey、_key方法,找到了就调用方法获取返回值。
没找到则查看accessInstanceVariablesDirectly方法的返回值,返回值为NO则调用valueForUndefinedKey:并抛出异常NSUnknownKeyException
; 返回值为YES则,顺序查找成员变量_key,_isKey,key,isKey,找到了直接取值,没找到的话也是调用valueForUndefinedKey:并抛出异常NSUnknownKeyException

面试题
  1. 什么是KVO
  • KVO是Key-value observing 的缩写
  • KVO是Objective-C对观察者设计模式的又一实现
  • Apple使用了isa混写(isa-swizzling)来实现KVO
  1. isa混写技术在KVO中是怎么体现的?
    当我们在调用 addObserver:forKeyPath时,系统会在运行时动态创建NSKVONotifying_A类,同时将原来的类指针指向新创建的类。新创建的NSKVONotifying_A类是原来类的子类,目的是重写setter方法,新的setter加入了通知所有观察对象的代码。
    调用了NSSetIntValueAndNotify
    实现如下:
- (void)setValue:(id)obj {
    [self willChangeValueForKey:@"keyPath"];
    [super setValue:obj];
    [self didChangeValueForKey:@"keyPath"];
}
  1. 通过kvc设置value,KVO能否生效
    能生效,kvc最终也是使用setter方法来赋值的。

  2. 通过成员变量直接赋值value,kvo能否生效?如果不能生效,如何让其生?
    直接给成员变量赋值不走setter,因而KVO不会生效。要使其生效,除了改成访问属性之外,还可以在成员变量赋值的前后加上[self willChangeValueForKey:@"value"][self didChangeValueForKey:@"value"]手动触发使其生效。

  3. KVC 的取值过程和原理是什么?

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值