小码哥底层原理笔记:Runtime之isa指针

前面我们知道在arm64位之后实例对象的isa指针&ISA_MASK之后是指向class类对象,class类对象的isa指针&ISA_MASK之后指向meta-class元类对象。在arm64位之前isa指针就是单纯的指向class类对象或者meta-class元类对象


isa指针

我们知道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对象的内存地址

你可能感兴趣的:(小码哥底层原理笔记:Runtime之isa指针)