KVC 底层原理
使用KVC,我们可以这样访问
CYXModel
的属性-
取值:
CYXModel *model = [[CYXModel alloc]init]; NSString *name = [model valueForKey: @"name" ]; NSString *productName = [model valueForKeyPath: @"product.productName" ];
设值:
CYXModel *model = [[CYXModel alloc]init];
[model setValue:@"CYX" forKey:@"name"];
[model setValue:@"NIKE" forKeyPath:@"product.productName"];
KVC字典转模型的实现原理
当调用setValue:forKey:设置属性value时,其底层的执行流程为:
【第一步】首先寻找是否有这三种
setter
方法,按照查找顺序为set<Key>:
->_set<Key>
->setIs<Key>
\1. 如果有其中任意一个setter
方法,则直接设置属性的value
(注意:key是指成员变量名,首字符大小写需要符合KVC的命名规范)
\2. 如果都没有,则进入【第二步】-
【第二步】:如果没有第一步中的三个简单的setter方法,则查找
accessInstanceVariablesDirectly
是否返回YES
。- 如果返回YES,则查找间接访问的实例变量进行赋值,查找顺序为:
_<key>
->_is<Key>
-><key>
->is<Key>
i. 如果找到其中任意一个实例变量,则赋值
ii. 如果都没有,则进入【第三步】
2.如果返回NO,则进入【第三步】
- 如果返回YES,则查找间接访问的实例变量进行赋值,查找顺序为:
【第三步】如果setter方法 或者 实例变量都没有找到,系统会执行该对象的
setValue:forUndefinedKey:
方法,默认抛出NSUndefinedKeyException
类型的异常
以下为以LGPerson的name属性为例,关于setter的流程图:
当调用valueForKey:时,其底层的执行流程如下:
- 【第一步】首先查找
getter
方法,按照get<Key>
-><key>
->is<Key>
->_<key>
的方法顺序查找,- 如果找到,则进入【第五步】
- 如果没有找到,则进入【第二步】
- 【第二步】如果第一步中的getter方法没有找到,KVC会查找
countOf <Key>
和objectIn <Key> AtIndex :
和<key> AtIndexes :
- 如果找到countOf <Key>和其他两个中的一个,则会创建一个响应所有NSArray方法的集合代理对象,并返回该对象,即NSKeyValueArray,是NSArray的子类。代理对象随后将接收到的所有
NSArray消息转换为
countOf<Key>,
objectIn<Key> AtIndex:和
<key>AtIndexes:消息的某种组合,用来创建键值编码对象。如果原始对象还实现了一个名为
get<Key>:range:`之类的可选方法,则代理对象也将在适当时使用该方法(注意:方法名的命名规则要符合KVC的标准命名方法,包括方法签名。) - 如果没有找到这三个访问数组的,请继续进入【第三步】
- 如果找到countOf <Key>和其他两个中的一个,则会创建一个响应所有NSArray方法的集合代理对象,并返回该对象,即NSKeyValueArray,是NSArray的子类。代理对象随后将接收到的所有
- 【第三步】如果没有找到上面的几种方法,则会同时查找countOf <Key>,enumeratorOf<Key>和memberOf<Key>这三个方法
- 如果这三个方法都找到,则会创建一个响应所有NSSet方法的集合代理对象,并返回该对象,此代理对象随后将其收到的所有NSSet消息转换为countOf<Key>,enumeratorOf<Key>和memberOf<Key>:消息的某种组合,用于创建它的对象
- 如果还是没有找到,则进入【第四步】
- 【第四步】如果还没有找到,检查类方法
InstanceVariablesDirectly
是否YES,依次搜索_<key>
,_is<Key>
,<key>
或is<Key>
的实例变量- 如果搜到,直接获取实例变量的值,进入【第五步】
- 【第五步】根据搜索到的属性值的类型,返回不同的结果
- 如果是对象指针,则直接返回结果
- 如果是
NSNumber支持的标量类型
,则将其存储在NSNumber实例
中并返回它 - 如果是是
NSNumber不支持的标量类型
,请转换为NSValue对象并返回该对象
- 【第六步】如果上面5步的方法均失败,系统会执行该对象的
setValue:forUndefinedKey:
方法,默认抛出NSUndefinedKeyException
类型的异常
以获取LGPerson的对象person的属性name为例,如下图的getter的流程图:
[图片上传失败...(image-157da3-1617109791958)]
KVO
键值观察者 (Key-Value Observer):是苹果提供的一套事件通知机制。允许对象监听另一个对象特定属性的改变,并在改变时接收到事件。
简单的说就是:观察者A监听被观察者B的某个属性,当B的属性发生更改时,A就会收到通知,执行相应的方法。
实现原理
[图片上传失败...(image-ea3878-1617109791958)]
基本的原理:当观察对象A时,KVO机制动态创建一个对象A的子类 NSKVONotifying_A,该类继承自对象A,并为这个新的子类重写观察属性keyPath的setter 方法。setter 方法会负责在调用原 setter 方法之前和之后,通知所有观察对象属性值的更改情况。在这个过程,被观察对象的 isa 指针从指向原来的A类,被KVO机制修改为指向系统新创建的子类NSKVONotifying_A类,来实现当前类属性值改变的监听
[图片上传失败...(image-c2d994-1617109791958)]