内存优化之ISA是什么?

2021-06-14  本文已影响0人  大王叫我来巡山丨

大家通常是否会认为isa就是对象的指针,用来表明对象所属的类型。
但是如果isa指针仅表示类型的话,对内存显然也是一个极大的浪费。于是,就像tagged pointer一样,对于isa指针,苹果同样进行了优化。使得isa指针表示的内容变得更为丰富,除了表明对象属于哪个类之外,还附加了引用计数extra_rc,是否有被weak引用标志位weakly_referenced,是否有附加对象标志位has_assoc等信息。接下来我们一起来看看它的真面目!

源码解析

首先,我们回顾一下isa指针是怎么在一个对象中存储的。如下图:


isa存储.png

从图中可以看出,我们所谓的isa指针,最后实际上落脚于isa_t的联合体;

isa_t分析

union isa_t {
    // 两个构造函数
    isa_t() { }
    isa_t(uintptr_t value) : bits(value) { }

    Class cls;
    // bits 可以当做为向外提供了操作struct的对象
    uintptr_t bits;
#if defined(ISA_BITFIELD)
    struct {
        ISA_BITFIELD;  // defined in isa.h
    };
#endif
};

从源码可以看出 isa_t是联合体,isa_t 中包含有cls,bits, struct三个变量,它们的内存空间是重叠的。在实际使用时,仅能够使用它们中的一种,你把它当做cls,就不能当bits访问,你把它当bits,就不能用cls来访问!(问:为什么用联合体呢,它的作用是什么?答:联合体的作用是在于,用更少的空间来表示了更多的类型,但是这些类型是不能够共存的!)
对联合体有兴趣的小伙伴空闲时间,可以去了解一下,但不建议用于项目中,装逼请随意!敲敲我这三米长的开山刀,请将注意力集中在isa_t 联合上:

这里需要注意下,虽说是三个成员,uintptr_t bits 和 struct 其实是一个成员,它们都占据64位内存空间!因为联合体类型的成员内存空间是重叠的。由于uintptr_t bits 和 struct 都是占据64位内存,因此它们的内存空间是完全重叠的!(简单的来说,uintptr_t bits 可以当做为向外提供了操作struct的对象,而struct 本身则说明了uintptr_t bits 中各个二进制位的定义)

理解了uintptr_t bits 和 struct 关系后,则isa_t其实可以看做有两个可能的取值,Class cls或struct。如下图所示:


isa_t.png

当isa_t作为Class cls使用时,这符合了我们之前一贯的认知:isa是一个指向对象所属Class类型的指针。然而,让一个64位的指针表示一个类型,是不是有些不划算呢。因此,绝大多数情况下,苹果采用了优化的isa策略,isa_t 类型并不等同于Class cls,而是struct!

一起来看一下struct的结构 :

# if __arm64__
#   define ISA_MASK        0x0000000ffffffff8ULL
#   define ISA_MAGIC_MASK  0x000003f000000001ULL
#   define ISA_MAGIC_VALUE 0x000001a000000001ULL
#   define ISA_BITFIELD                                                      \
      uintptr_t nonpointer        : 1;                                       \  注意:区分isa_t是否是一个真正的指针!!!
      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
#   define RC_ONE   (1ULL<<45)
#   define RC_HALF  (1ULL<<18)

# elif __x86_64__
#   define ISA_MASK        0x00007ffffffffff8ULL
#   define ISA_MAGIC_MASK  0x001f800000000001ULL
#   define ISA_MAGIC_VALUE 0x001d800000000001ULL
#   define ISA_BITFIELD                                                        \
      uintptr_t nonpointer        : 1;                                         \
      uintptr_t has_assoc         : 1;                                         \
      uintptr_t has_cxx_dtor      : 1;                                         \
      uintptr_t shiftcls          : 44; /*MACH_VM_MAX_ADDRESS 0x7fffffe00000*/ \
      uintptr_t magic             : 6;                                         \
      uintptr_t weakly_referenced : 1;                                         \
      uintptr_t deallocating      : 1;                                         \
      uintptr_t has_sidetable_rc  : 1;                                         \
      uintptr_t extra_rc          : 8
#   define RC_ONE   (1ULL<<56)
#   define RC_HALF  (1ULL<<7)

# else
#   error unknown architecture for packed isa
# endif

// SUPPORT_PACKED_ISA
#endif

struct共占用64位,从低位到高位依次是nonpointer到extra_rc。成员后面的:表明了该成员占用几个bit。成员的含义如下:


struct 结构表.png

由上表可以看出,和对象引用计数相关的有两个成员:extra_rc和has_sidetable_rc。真机所采用19位的extra_rc来记录对象的引用次数,当extra_rc 不够用时,还会借助sidetable来存储计数值,这时,has_sidetable_rc会被标志为1。我们可以算一下,对于19位的extra_rc ,其数值可以表示2^19 - 1 = 524287。 52万多,相信绝大多数情况下,都够用了。

实践验证

创建一个对象,并打印isa_t的值

PeopleModel *model = [[PeopleModel alloc] init];
NSLog(@"isa_t = %p", *(void **)(__bridge void*)model); // isa_t = 0x1a10286cf39

复制该地址粘贴到计算机中:


计算器.png

标红区域请于上图中struct结构表结合查看!当开启了isa_t优化,nonpointer 置位为1, 这时,isa_t *其实不是一个地址,而是一个有意义的值,也就是说,苹果用isa_t * 所占用的64位空间,表示了一个有意义的值,而这64位值的定义,就符合我们上面struct的定义。

上一篇 下一篇

猜你喜欢

热点阅读