今天做了一下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。一般情况下打补丁没有这个需求。