KVC & KVO原理

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:

你可能感兴趣的:(KVC & KVO原理)