美文网首页iOS开发iOS开发iOS技术点
class_addProperty方法中的objc_proper

class_addProperty方法中的objc_proper

作者: 千若逸 | 来源:发表于2016-04-05 17:12 被阅读1648次

    今天做了一下runtime相关的东西,踩了一个坑,记录一下。

    先看这段代码:

    + (void)addStrPropertyForTargetClass:(Class)targetClass Name:(NSString *)propertyName{
        objc_property_attribute_t type = { "T", [[NSString stringWithFormat:@"@\"%@\"",NSStringFromClass([NSString class])] UTF8String] }; //type
        objc_property_attribute_t ownership0 = { "C", "" }; // C = copy
        objc_property_attribute_t ownership = { "N", "" }; //N = nonatomic
        objc_property_attribute_t backingivar  = { "V", [[NSString stringWithFormat:@"_%@", propertyName] UTF8String] };  //variable name
        objc_property_attribute_t attrs[] = { type, ownership0, ownership, backingivar };
        if (class_addProperty(targetClass, [propertyName UTF8String], attrs, 4)) {
            //添加get和set方法
            [targetClass addObjectProperty:propertyName];
            DDLogDebug(@"创建属性Property成功");
        }
    }
    

    上面的代码可以在运行时给一个类添加新的NSString对象,这个对象的一些主要特性主要取决于代码中的objc_property_attribute_t定义。

    由于动态配置的需要,我需要对objc_property_attribute_t变量进行灵活配置,因为objc_property_attribute_t就是一个二值结构体,所以很适合用Dictionary来储存,于是我采用了Dictionary来储存所有objc_property_attribute_t变量的字符串信息。

    之后再遍历Dictionary,取得attribute数据并添加进attrs数组中。

    问题这时候就出现了,在随后我需要获取这个新property的type时,居然是空的。

    随后我用下面的代码打印所有属性:

    @autoreleasepool {
            unsigned int outCount, i;
            objc_property_t *properties = class_copyPropertyList([TestClass class], &outCount);
            for (i = 0; i < outCount; i++) {
                objc_property_t property = properties[i];
                fprintf(stdout, "mytest %s %s\n", property_getName(property), property_getAttributes(property));
            }
        }
    

    结果为:

    mytest strNew V_strNew,T@"NSString",C,N
    mytest strName T@"NSString",C,N,V_shopid
    

    其中的strName就是原生Property变量,strNew是新添加的。可以发现strNew和strName后面的属性有些细微区别:顺序不一样——这就是问题原因?
    抱着试一试的态度,我就真的改变了新代码中添加attribute的顺序,与strName的attribute顺序保持一致,再测试,居然就好了。

    说明这个顺序不是随便设置的,后面在github上的这份代码CocoaScript/MOProtocolDescription.m at master · ccgus/CocoaScript里找到一些说明,其中特别指出Type encoding must be first,Backing ivar must be last。

    2017年5月13日下午4:39
    今天见有人问这个用途在哪里,既然有人问起,那就说一下吧,是用于JSPatch上的。JSPatch中用它自带的方法添加的属性是不支持反射的,我记得好像是通过JS的方法添加的,就是说没有走OC的底层来添加属性,这导致MJExtension,JsonModel等第三方JSON处理库没办法处理这种通过js添加的属性,因为它们是通过OC的反射技术来遍历一个类或者对象的所有属性的。

    相关的开源库:wonderffee/DFDynamicProperty
    可能有内存泄露,慎用

    原理

    用到了class_addProperty,这个可以给已经存在的类添加新的property,而且能够用过反射遍历到动态添加的属性。正好适合于JSPatch打补丁的情况
    之所以不通过调用class_addIvar来添加实例变量,是因为它会改变一个已有类的内存布局,一般是通过objc_allocateClassPair动态创建一个class,才能调用class_addIvar创建Ivar,最后通过objc_registerClassPair注册class。一般情况下打补丁没有这个需求。

    参考:

    ios动态添加属性的几种方法 - shengyumojian的专栏 - 博客频道 - CSDN.NET

    相关文章

      网友评论

      • 目前运行时:说的可以 兄弟!
      • 老板娘来盘一血:作者你好,我在练习 runtime 的时候发现:打印已有类的成员变量和属性时,属性自动生成了对应的下划线成员变量名(这里没有任何问题);但是当我打印动态生成的类的成员变量和属性时,发现动态添加的属性并没有生成对应的成员变量名(具体来说 成员变量列表里并没有属性应该对应的那个下划线成员变量),不知道你有木有遇到这个问题,还望请教一下
      • 穿山甲到底说了什么:Ivar ivar = class_getInstanceVariable(class, [[NSString stringWithFormat:@"_%@", propertyName] UTF8String]);
        即使property添加成功,可 ivar 结果仍然为空,...
      • e8d19cc9c82f:在哪种场景下要使用runtime来添加类的属性,起到什么作用?能解答一下下吗?谢谢
        千若逸:我是用在JSPatch中的,JSPatch中用它的方法添加的属性是不支持反射的,也就没办法用MJExtension,JsonModel等第三方json处理库

      本文标题:class_addProperty方法中的objc_proper

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