1、KVC
KVC
可以通过key
直接访问对象的属性,或者给独享的属性直接赋值,这样可以在运行时动态的访问或修改对象的属性。
当调用setValue:forKey:
setValue:forKeyPath:
时,底层的执行机制如下:
1. 程序首先按照`setKey:`、`_setKey:`、`setIsKey:`顺序查找方法,如果找到直接调用该方法,结束。
2. 如果没有找到,则会查看`+(BOOL)accessInstanceVariablesDirectly`方法的返回值:
2.1. 如果返回`NO`,表示不允许访问成员变量,则直接执行`setValue:forUndefinedKey:`方法抛出异常,结束。
2.2. 如果返回`YES`,表示允许访问成员变量。
3. 如果允许访问成员变量,则按照 `_key`、`_isKey`、`key`、`isKey` 的顺序查找成员变量,如果找到了成员变量,则直接赋值,结束。
4. 如果没有仍然没有找到成员变量,则直接执行`setValue:forUndefinedKey:`方法抛出异常,结束。
当调用valueForKey:
、valueForKeyPath:
时,底层的执行机制如下:
1. 程序首先按照`getKey`、`key`、`isKey`、`_getKey`、`_key`顺序查找方法,如果找到直接调用该方法,结束。
2. 如果没有仍然没有找到以上方法,则会调用一些动态生成的集合方法(这些方法可以可重新定义KVC的一些功能):
2.1. 如果同时存在`countOf`和[`objectInAtIndex:` `AtIndexes:`]中的一个方法,
则返回一个可以响应NSArray所有方法的代理集合
还有一个可选方法 `get:range:`
2.2. 如果同时存在 `countOf`,`enumeratorOf`,`memberOf:` 三个方法,
则返回一个可以响应NSSet所有方法的代理集合
3. 如果没有找到,则会查看`+(BOOL)accessInstanceVariablesDirectly`方法的返回值:
3.1. 如果返回`NO`,表示不允许访问成员变量,则直接执行`valueForUndefinedKey`方法抛出异常,结束。
3.2. 如果返回`YES`,表示允许访问成员变量。
4. 如果允许访问成员变量,则按照`_key`、`_isKey`、`key`、`isKey`的顺序查找成员变量,如果找到了成员变量,则直接赋值,结束。
5. 如果没有仍然没有找到成员变量,则直接执行`valueForUndefinedKey:`方法抛出异常,结束。
2、KVO
KVO
全称KeyValueObserving
,是苹果提供的一套事件通知机制。允许对象监听另一个对象特定属性的改变,并在改变时接收到事件。由于KVO的实现机制,所以对属性才会发生作用,一般继承自NSObject的对象都默认支持KVO
实现原理
-
KVO
是通过isa-swizzling
技术实现的。 - 在运行时根据原类创建一个中间类
NSKVONotifying_KVOObject
,这个中间类是原类的子类,并动态修改当前对象的isa指向中间类。并且将class方法重写,返回原类的Class。
在 - 重写被监听属性的
setter
方法:在重写的setter方法中,修改值之前会调用willChangeValueForKey:
方法,修改值之后会调用didChangeValueForKey:
方法,这两个方法最终都会被调用到observeValueForKeyPath:ofObject:change:context:
方法中
面试题
1、使用KVC
设值能触发KVO
吗?直接修改成员变量呢?
会触发KVO,KVC内部会调用 `willChangeValueForKey:` 和 `didChangeValueForKey:` 方法
直接修改修改成员变量不会触发 KVO
2、iOS用什么方式实现对一个对象的KVO?(KVO的本质是什么?)
当一个对象使用了KVO监听,iOS系统会修改这个对象的isa指针,改为指向一个全新的通过Runtime动态创建的子类
子类拥有自己的set方法实现,内部会调用
willChangeValueForKey:
原来的setter
didChangeValueForKey:, 这个方法的内部又会调用监听器observer的方法
3、如何手动出发KVO?
手动调用这两个方法,必须两个方法同时有
willChangeValueForKey:
didChangeValueForKey: