runtime可以以底层的角度来对一些实现方式进行更改,比如说KVC
首先,先来了解下KVC的底层原理:
key : value
- 1.去模型中查找有没有setValue:,直接调用这个对象setValue:赋值
- 2.如果没有setValue:,就在模型中查找_value属性
- 3.如果没有_value属性,就查找value属性
- 4.如果还没有就报错
在和后台通信的JSON数据中,可能会看到这种JSON数据:
1 | { |
其中的id是什么?是Objective-C关键字,也就是说我不能定义以下属性:
1 | @property (nonatomic, strong) NSString *id; |
由于数据模型名称没有和JSON的键值一一对应,我们不能使用以下方法,对模型中的成员变量进行统一设置:
1 | - (void)setValuesForKeysWithDictionary:(NSDictionary *)keyedValues; |
既然这样,可以选择手动一个个去实现。但是这样在数据少的时候可以试试,在数据比较多时就不太现实了,程序的可扩展性也不好。
现在来了解下相对比较简单的两种解决方法:
方式1.重写setValue:forKey:
setValuesForKeysWithDictionary:的底层是调用setValue:forKey:的,所以可以考虑重写这个方法,并且判断其key是id时,手动转换成模型的成员变量名,这里假设把id对应成以下属性:
1 | @property (nonatomic, strong) NSString *ID; |
有了对应的属性名后,就可以重写底层方法了:
- 如下所示,当判断到key的值为id时,我手动将key转换成了模型属性名,即ID
1 | - (void)setValue:(id)value forKey:(NSString *)key |
这样,当使用setValuesForKeysWithDictionary:就不会出现模型中找不到对应的成员变量的错误了。
方式2.使用runtime
考虑到runtime和KVC的实现原理,可以使用另一种实现思路,就是先在模型中找到对应的成员变量,然后从JSON字典中找到对应的数据进行赋值
。
这里先要了解runtime的两个实例变量操作方法:
1 | // 获取成员变量列表 |
详细实现步骤:
- 1.获取模型中的所有实例变量
1 | unsigned int outCount = 0; |
- 2.将获取出来以’‘开头的实例变量名去处’‘符号
1 | NSString *ivarName = @(ivar_getName(ivar)); |
- 3.获取JOSN字典中对应的value,如果没有,手动设置我们传入的字典映射,以指定对应的模型变量名,最后调用setValue:forKeyPath:设置模型实例变量值
1 | id value = dict[ivarName]; |
由于需要针对所有模型使用,可以将其设置为NSObject分类。以上步骤的完整代码为:
1 | // dict -> 资源文件提供的字典 |
使用方法:
1 |
|
看了一些相关框架的源码,有些框架的底层就是通过这种方式实现的,比如MJExtension就是通过获取对象里面的所有属性来进行操作的(这里个人感觉获取成员变量适用面会更广一点)