在程序设计中,为了使我们所写的程序最大程度的满足更多要求,又要满足程序设计的低耦合性,一种可能的方案可能在很多工程师的脑海里翻滚过:如果我能够动态的创建类,动态给类添加方法属性该有多好。
ok,我们用如下方法可以做到简单的处理:
Class class = objc_allocateClassPair(NSObject.class, "Company", 0);
class_addIvar(class, "_man", sizeof(id), log2(sizeof(id)), @encode(id));
class_addIvar(class, "_woman", sizeof(id), log2(sizeof(id)), @encode(id));
class_addIvar(class, "_boss", sizeof(id), log2(sizeof(id)), @encode(id));
objc_registerClassPair(class);
id com = [class new];
Ivar weakIvar = class_getInstanceVariable(class, "_woman");
Ivar strongIvar = class_getInstanceVariable(class, "_man");
//{
id girl = [NSString stringWithFormat:@"girl"];
id boy = [NSObject new];
object_setIvar(com, weakIvar, girl);
object_setIvar(com, strongIvar, boy);
//}
// 输出:weakIvar 为 nil,strongIvar 有值
NSLog(@"%@, %@", object_getIvar(com, weakIvar), object_getIvar(com, strongIvar));
好的,这段代码会没有任何问题的执行出来,但是,这段代码要正确执行,有个前提是那个{}被注释掉,如果放开之后会如何?
没错放开后会出现两种情况:
A arc为yes情况下,会出现bad access
B arc为No情况下,正常打印出结果
怎么样解决这个问题?出现这个问题的原因又是为什么?
出现这个问题的一个原因是:正常编译的类,且标识了 -fobjc-arc flag 时,这个标记位为 1,而动态创建的类并没有设置它。
所以解决这个问题的一种可行方案是标注他:因为class是个数据结构,所以要设置,就要转换,转换的最直接办法就是内存拷贝,所以可以有以下方法:
static void fix_up_class_arc(Class class) {
struct {
Class isa;
Class superclass;
struct {
void *_buckets;
uint32_t _mask;
uint32_t _occupied;
} cache;
uintptr_t bits;
} *objcClass = (__bridge typeof(objcClass))class;
if !LP64
define FAST_DATA_MASK 0xfffffffcUL
else
define FAST_DATA_MASK 0x00007ffffffffff8UL
endif
struct {
uint32_t flags;
uint32_t version;
struct {
uint32_t flags;
} *ro;
} *objcRWClass = (typeof(objcRWClass))(objcClass->bits & FAST_DATA_MASK);
define RO_IS_ARR 1<<7
objcRWClass->ro->flags |= RO_IS_ARR;
}
所以只要能够有这段代码就可以解决arc下的问题
objc_registerClassPair(class);
fix_up_class_arc(class);
id com = [class new];