美文网首页
利用RunTime做一点点代码优化记录

利用RunTime做一点点代码优化记录

作者: 杜员外 | 来源:发表于2019-07-05 15:54 被阅读0次

    现状

    业务这边之前有需求各种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或者其他的工具解析就完事了。我这边因为是上古代码的关系,不可能直接修改改动太大,所以才出此下策。

    大概思路

    1. 后端或者本地缓存拿到数据。
    2. 如果是QA环境需要检查本地的debug数据,看是否设置过本地的config,如果设置过则需要将debug数据更新上去。
    3. 通过便利拿到的数据通过key值通过runtimeobject_setIvar一一对应的设给对应的Model。

    难点一

    1. 代码里面的属性名称跟后端的Key对不上。
      比如,后端的key是TestKey,但是代码里面的属性名称是1111。(只是比喻)
    2. 有一些config在设计的时候不是按照一个规范来的。需要对这个config做一个特殊的处理。

    难点解决

    1. 照着Mantle的模式写了一个方法返回一个字典。对应的Key、Value分别是后端的Key以及本地的属性名称。
    2. 另外一个方法,返回的是需要特殊处理的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);
        }
    }
    

    后续继续跟进结果。如果有问题会继续更新

    相关文章

      网友评论

          本文标题:利用RunTime做一点点代码优化记录

          本文链接:https://www.haomeiwen.com/subject/hxeehctx.html