前面我们知道在arm64位之后实例对象的isa指针&ISA_MASK之后是指向class类对象,class类对象的isa指针&ISA_MASK之后指向meta-class元类对象。在arm64位之前isa指针就是单纯的指向class类对象或者meta-class元类对象
我们知道OC对象的本质就是一个结构体在arm64位之后是这样:
struct objc_object {
isa_t isa;//isa指针
}
发现这个isa指针是一个isa_t结构体:
union isa_t
{
Class cls;
uintptr_t bits;
struct {
uintptr_t nonpointer : 1;
uintptr_t has_assoc : 1;
uintptr_t has_cxx_dtor : 1;
uintptr_t shiftcls : 33; // MACH_VM_MAX_ADDRESS 0x1000000000
uintptr_t magic : 6;
uintptr_t weakly_referenced : 1;
uintptr_t deallocating : 1;
uintptr_t has_sidetable_rc : 1;
uintptr_t extra_rc : 19;
};
}
isa_t是一个共用体,并且使用位域来存储更多信息。注:(通过bits&XX_MASK可以取出位域中对应的值)
(1)nonpointer:0代表普通的指针,存储着Class、Meta-Class对象的内存地址;1代表优化过,使用位域存储更多信息
(2)has_assoc:是否有设置过关联对象,如果没有,释放时会更快
(3)has_cxx_dtor:是否有C++的析构函数,如果没有,释放时会更快
(4)shiftcls:存储着Class、Meta-Class对象的内存地址信息
(5)magic:用于调试时分辨对象是否未完成初始化
(6)weakly_referenced:是否有被弱引用指向过,如果没有,释放时会更快
(7)deallocating:对象是否正在释放
(8)extra_rc:里面存储的值是引用计数值
(9)has_sidetable_rc:引用计数器是否过大无法存储在isa中,如果为1,那么引用计数会存储在一个叫SideTable的类的属性中
所以isa& ISA_MASK其实就是取出位域中shiftcls的值,由此得到Class、Meta-Class对象的内存地址