swift 基础

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

2022-03-28  本文已影响0人  十拿九稳啦

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有

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
  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 的取值过程和原理是什么?

上一篇下一篇

猜你喜欢

热点阅读