美文网首页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

相关文章

  • class_addProperty方法中的objc_proper

    今天做了一下runtime相关的东西,踩了一个坑,记录一下。 先看这段代码: 上面的代码可以在运行时给一个类添加新...

  • class_addProperty运行时方法

    static char *propertykey = nil; NSString *getterMeth(id s...

  • 01.11 - 类的方法–对象方法

    class 类名: 类中的属性 类中的方法 类中的方法 声明在类中的函数就是方法 类中的方法包括:对象方法(实例方...

  • Number中的方法

    Number中的方法 构造函数1 Number(value)2 new Number(value) 属性方法1 N...

  • Swift中的方法

    结构体和美剧能够定义方法是Swift和C/OC的主要区别之一。 1. 实例方法 实例方法是属于某个特定类、结构体或...

  • swift中的方法

    方法是关联了特定类型的函数。类,结构体以及枚举都能定义实例方法,方法封装了给定类型特定的任务和功能。类,结构体和枚...

  • 唱歌中的方法

    学习音乐不是让每个人都成为名家,也不一定是要走音乐专业的道路,而是一种艺术修养的拥有。重庆图兰朵声乐中心是重庆最专...

  • java中的方法

    所谓方法,就是用来解决一类问题的代码的有序组合,是一个功能模块。 1.一般情况下,定义一个方法的语法是: 2.分析...

  • Swift 中的方法

    实例方法 方法的外部参数名 默认情况下,方法的第一个参数只有局部参数名,而第二个开始,都有一个外部参数名。(与 O...

  • IOS中的方法

    在IOS中系统的方法名较长,但是易读,例如: 自定义方法: 有1个方法的方法名-(void)setImage:(N...

网友评论

  • 目前运行时:说的可以 兄弟!
  • 老板娘来盘一血:作者你好,我在练习 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