现状
业务这边之前有需求各种config,从后端拿到数据,一个字典挨个解析。
最早的写法如下:
...
NSNumber* xxxx = [infoDic objectForKey:@"xxxx"];
if (xxxx != nil && [xxxx isKindOfClass:[NSNumber class]]) {// or [NSString class]
int nRateDialog = [xxxx intValue];
if (nRateDialog == 1) {
infoModel.xxxx = YES;
}else{
infoModel.xxxx = NO;
}
}
** 很多类似的写法
...
代码臃肿不说,很丑。
但是,因为上古版本的原因,类中定义的数据类型也是随心所欲。
int NSInteger NSString NSNumber BOOL float CGFloat
几乎包含了所有的能用到的类型...
再加上这边要给QA做一个开关的需求,本地随便修改这些config。所有有了这篇记录。
当然如果你本身代码很规范,直接用Mantle或者其他的工具解析就完事了。我这边因为是上古代码的关系,不可能直接修改改动太大,所以才出此下策。
大概思路
- 后端或者本地缓存拿到数据。
- 如果是QA环境需要检查本地的debug数据,看是否设置过本地的config,如果设置过则需要将debug数据更新上去。
- 通过便利拿到的数据通过key值通过runtime的object_setIvar一一对应的设给对应的Model。
难点一
- 代码里面的属性名称跟后端的Key对不上。
比如,后端的key是TestKey,但是代码里面的属性名称是1111。(只是比喻) - 有一些config在设计的时候不是按照一个规范来的。需要对这个config做一个特殊的处理。
难点解决
- 照着Mantle的模式写了一个方法返回一个字典。对应的Key、Value分别是后端的Key以及本地的属性名称。
- 另外一个方法,返回的是需要特殊处理的key。
这样写的好处是便于后续扩展。
如果真的需要两端不相同,只需在1方法中加上对应的Key、Value就行。如果需要特殊处理只需在2加对应的值就行。并且,后续添加config只需.h 方法中新增就行,无需多加代码,提高效率。
难点二
在写代码过程中,发现runtime的方法object_setIvar设值的问题。
object_setIvar(id _Nullable obj, Ivar _Nonnull ivar, id _Nullable value)
object_setIvar 需要传三个值。 obj是你要赋值的对象。ivar是你通过属性名拿到的指针。value是你要设置的值
无论你是百度还是Google,你搜到的runtime相关教程在这里设置的都是一个string或者其他对象。但是int,NSInteger,float或者CGFloat它不是一个对象,如果你直接设置就会报错。
所以又是一番查找,最后发现了解决方式。下面是stackoverflow链接。
stackoverflow
直接通过指针投射过去! NB
至此,所有难点就攻克。代码什么的随便写写就完事,重点是思路以及runtime相关知识。
更重要的是把之前快1000行的垃圾代码优化到了200多行,如果所有的规范统一的话100行甚至更少
runtime这边拿到的属性值。需要记录一下
BOOL -> TB
int -> Ti
NSNumber -> T@"NSNumber"
NSString -> T@"NSString"
NSInteger -> Tq
NSArray -> T@"NSArray"
id -> T@
float -> Tf
CGFloat -> Td
如果属性是一个枚举(NS_ENUM),那么你拿到的是你枚举对应的值。
比如:
typedef NS_ENUM(NSInteger, TestStyle) {
};
拿到的就是Tq。
7-9 Update:
void (*object_setIvarFloat)(id, Ivar, CGFloat) = (void (*)(id, Ivar, CGFloat))object_setIvar;
使用这种方式莫名的会出现赋值无法成功的情况。
最终还是选择了如果是int这种情况直接采用拿对应属性的地址偏移量直接赋值,下面是代码:
void setIvarMethod1 (const char * type, id n, Ivar ivar, NSString *obj) {
ptrdiff_t offset = ivar_getOffset(ivar);
unsigned char *stuffBytes = (unsigned char *)(__bridge void *)n;
if (strcmp(type, "B") == 0) {
*((BOOL *)(stuffBytes + offset)) = obj.boolValue;
}else if (strcmp(type, "q") == 0){
*((NSInteger *)(stuffBytes + offset)) = obj.integerValue;
}else if (strcmp(type, "i") == 0) {
*((int *)(stuffBytes + offset)) = obj.intValue;
}else if (strcmp(type, "d") == 0 || strcmp(type, "f") == 0){
*((float *)(stuffBytes + offset)) = obj.floatValue;
}else{
object_setIvar(n, ivar, obj);
}
}
后续继续跟进结果。如果有问题会继续更新
网友评论