美文网首页iOS底层原理
小码哥底层原理笔记:Runtime之isa指针

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

作者: chilim | 来源:发表于2020-07-01 17:32 被阅读0次

    前面我们知道在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指针

        本文链接:https://www.haomeiwen.com/subject/ttchqktx.html