美文网首页
[Objective-C-Runtime系列] isa指针详解

[Objective-C-Runtime系列] isa指针详解

作者: coder_clownfish | 来源:发表于2019-01-22 11:29 被阅读0次

    学习runtime涉及知识点,首先需要学习的是runtime的对象模型以及整体架构,对runtime整体有了基本的了解之后,你会发现他的API实现思路其实并不是特别复杂,
    另外,由于本人学习底层知识的途径是李明杰老师的教学视频,所以大部分内容会有相同或者相似的地方,文章中也会尽量多的涉及自己的思考.下面进入今天的正题

    isa指针

    首先什么是isa指针?为什么学习runtime需要先学习isa指针呢?
    在oc中,大部分的对象都是继承自NSObject,NSObject底层简化结构为

    struct objc_object {
    private:
        isa_t isa;
    }
    

    按照我的个人理解,就是对象本身寻找自己所属类的途径,你可以理解为你本身为一个实例对象.你的isa指针就会指向所属类对象->中国人,而中国人类对象又会通过他的isa指针指向他所属的类对象亚洲人,按照一层一层的关系最终会寻找到最基础的类,成为元类(metaClass).
    对isa说明最经典的还是网上的一张图片

    Snip20190122_1.png
    第一次看这张图时,一脸懵逼,这是什么?
    其实这就是当你创建一个实例对象之后,实例对象查找类对象和元类对象的路线图,以及通过superClass指针查找父类的路线图,
    isa指针是横向的关系,从实例对象instance object isa -> class object isa -> meta class.这些都是上层的知识点.当你理解了isa指针是什么,并且了解了他的作用之后,关注点会变更为oc实现isa指针的原理逻辑

    isa底层实现

    isa底层结构经历过优化,时间点为arm64架构之后
    arm64架构之前,isa指针知识单纯地一个C指针,指向Class对象/metaClass对象.
    arm64架构之后,isa指针使用了isa_t结构.在获取存储Class内存地址时不能直接使用isa,需要做位运算 isa & ISA_MASK(0x0000000ffffffff8ULL)

    struct objc_object {
                isa_t isa;
            }
    

    isa_t简化结构如下,我去除了条件编译的部分,只拿出arm64 架构下的结构讲解,其他架构是一样的思路.

    union isa_t {
                uintptr_t bits;
    
                struct {
                    uintptr_t nonpointer        : 1; // 是否是指针,旧架构isa为指针,值为0, 新架构为isa_t结构体,值为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; // 引用计数器是否过大,如果为1,引用计数回存储在一个SlideTable的类属性中
                    uintptr_t extra_rc          : 19; // 引用计数-1
                };
            }
    

    isa_t使用的是共用体以及位域结构实现的,对位域和共用体不了解的小伙伴可以自己学一下,这里我只介绍大概使用.

    共用体:就是所有的数据共用一块内存空间,例如

    union cf_test {
        int cf1;
        int cf2;
        int cf3;
    }
    
    cf_test.cf2 = 2;
    NSLog(@"%d", cf_test.cf3) == 2;
    

    以上结构中,当你更改了任何一个变量,cf_t.cf1 = 1.0,调用其他的变量都会获取到值,因为共用的是同一块内存,
    位域:指信息在存储时,并不需要占用一个完整的字节, 而只需占几个或一个二进制位.示例如下

    struct foo_t {
      unsigned int b1 : n1,
      b2 : n2,
      bn : nk,
    };
    b1,b2,b3是字段明,可以通过名称访问,声明时需要注明类型.
    n1,n2,nk等代表的是bit数.
                    
    

    下面在看isa_t的结构会不会觉得就理解了,isa_t结构本身是一个共用体,内部变量bits 和 struct 共同使用一块内存, struct 结构体使用位域技术表明每段二进制位存储的内容.每个字段的含义在上文中已经添加过注释.
    想获取每段二进制位存储的内容,只需要对isa.bits做&运算获取,
    已经定义好的宏定义如下,

    #   define ISA_MASK        0x0000000ffffffff8ULL
    #   define ISA_MAGIC_MASK  0x000003f000000001ULL
    #   define ISA_MAGIC_VALUE 0x000001a000000001ULL
    

    如果对位运算获取存储内容不太清楚地同学,建议可以学习一下位运算知识,因为位运算在oc中也是很有应用的,例如平时常见的多项枚举类型的使用,我随便截取的一段代码做示例,options参数可以传递多个枚举值,使用的也是位运算的方式.

    [self.iconImageView sd_setImageWithURL:[NSURL URLWithString:urlString] placeholderImage:nil options:SDWebImageRetryFailed | SDWebImageAllowInvalidSSLCertificates];
    

    isa基本的知识点就这些了,由于第一次写博客,同时为更详细的讲解清楚isa指针,所以篇幅较长,同时逻辑有点不清晰,后面会逐渐改进,
    如果有错误欢迎指正,同时一直在寻找志同道合的道友共同进步,有喜欢交朋友的随时欢迎撩骚我啊.
    -谢

    相关文章

      网友评论

          本文标题:[Objective-C-Runtime系列] isa指针详解

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