《OC底层系列四》-isa&superclass分析

2020-10-13  本文已影响0人  002and001

前言

目录

目录.png

1、简介

isa&superclass.png

lldb指令复习

lldb指令复习.png

2、isa走位

在可编译运行调试的objc-781源码工程中,创建Person类

// Person.h
@interface Person : NSObject

@end

// Person.m
@implementation Person

@end

main.m添加断点如下:


image.png

我们使用lldb命令进行验证。

2.1、对象的isa指向类

// 获取p1的内存地址
(lldb) p/x p1
(Person *) $0 = 0x0000000102a042c0
// 打印p1的内存信息
(lldb) x/4gx p1
0x102a042c0: 0x001d8001000080f9 0x0000000000000000
0x102a042d0: 0x5376614e534e5b2d 0x7265536465726168
// 获取p1的isa指向的内存地址
(lldb) p/x 0x001d8001000080f9 & 0x00007ffffffffff8ULL
(unsigned long long) $2 = 0x00000001000080f8
// 打印p1的isa指向的内存地址description
(lldb) po 0x00000001000080f8
Person
// 打印Person.class的内存地址
(lldb) p/x Person.class
(Class) $4 = 0x00000001000080f8 Person
// 打印object_getClass(p1)的内存地址
(lldb) p/x object_getClass(p1)
(Class) $5 = 0x00000001000080f8 Person

  在上一篇《OC底层系列三》-对象和类的关联里我们知道,对象的类信息存储在其isa中的的shiftcls中,对象和类的通过对象中的ias的shiftcls进行关联,class方法本质上返回的是isa的shiftcls,其底层实现通过isa.bit& ISA_MASK(0x00007ffffffffff8ULL)获取到shiftcls的值。
  通常我们可以理解为类实例化的对象的isa指向该类

  isa走位分析结果1的lldb打印结果再次印证此过程;同时我们还知道,类的底层实际上是一个objc_class的结构体,继承自objc_object

2.2、类isa指向元类

前面我们得到Person类的内存地址为0x00000001000080f8
继续使用lldb进行调试分析如下图:

// 打印Person类的内存信息
(lldb) x/4gx 0x00000001000080f8
0x1000080f8: 0x00000001000080d0 0x0000000100333028
0x100008108: 0x0000000102a15c60 0x0001801000000003
// 获取Person类的isa指向的内存地址
(lldb) p/x 0x00000001000080d0 & 0x00007ffffffffff8ULL
(unsigned long long) $6 = 0x00000001000080d0
// 打印Person类的isa指向的内存地址的description
(lldb) po $6
Person

  通过isa走位分析2我们可以看到,Person类的isa还是指向一个Person,我们称之为Person元类
上面的过程用图描述如下:

image.png
和官方的isa指向一致。

2.3、元类的isa指向NSObject元类

// 打印Person元类的内存信息
(lldb) x/4gx 0x00000001000080d0
0x1000080d0: 0x0000000100333000 0x0000000100333000
0x1000080e0: 0x0000000102a161c0 0x0004e03100000007
// 获取Person元类的isa指向的内存地址
(lldb) p/x 0x0000000100333000 & 0x00007ffffffffff8ULL
(unsigned long long) $7 = 0x0000000100333000
// 获取Person元类的isa指向的内存地址的description
(lldb) po $7
NSObject

2.4、NSObject元类isa指向自身

// 打印NSObject元类的内存信息
(lldb) x/4gx 0x0000000100333000
0x100333000: 0x0000000100333000 0x0000000100333028
0x100333010: 0x00000001006b66d0 0x0004e03100000007
// 获取NSObject元类的isa指向的内存地址
(lldb) p/x 0x0000000100333000 & 0x00007ffffffffff8ULL
(unsigned long long) $8 = 0x0000000100333000
// 获取NSObject元类的isa指向的内存地址的description
(lldb) po $8
NSObject

2.5、NSObject类的isa指向NSObject元类

// 获取NSObject类的内存地址
(lldb) p/x NSObject.class
(Class) $9 = 0x0000000100333028 NSObject
// 打印NSObject类指向内存的信息
(lldb) x/4gx 0x0000000100333028
0x100333028: 0x0000000100333000 0x0000000000000000
0x100333038: 0x0000000102a16020 0x0001801000000003
// 获取NSObject类的isa指向的内存地址
(lldb) p/x 0x0000000100333000 & 0x00007ffffffffff8ULL
(unsigned long long) $10 = 0x0000000100333000
// 打印NSObject类isa的description
(lldb) po $10
NSObject

2.6、NSObject类的实例的isa指向NSObject类

// 获取01的内存地址
(lldb) p/x o1
(NSObject *) $11 = 0x0000000102a15ca0
// 打印o1内存信息
(lldb) x/4gx o1
0x102a15ca0: 0x001d800100333029 0x0000000000000000
0x102a15cb0: 0x0000000000000000 0x0000000000000000
// 获取o1的isa指向的内存信息
(lldb) p/x 0x001d800100333029 &0x00007ffffffffff8ULL
(unsigned long long) $13 = 0x0000000100333028
// 获取o1的isa指向的内存信息的description
(lldb) po $13
NSObject

和官方的isa指向一致。

3、superclass走位

// 打印NSObject类指向内存的信息
(lldb) x/4gx 0x0000000100333028
0x100333028: 0x0000000100333000 0x0000000000000000
0x100333038: 0x0000000102a16020 0x0001801000000003

// 打印NSObject元类指向内存的信息
(lldb) x/4gx 0x0000000100333000
0x100333000: 0x0000000100333000 0x0000000100333028
0x100333010: 0x00000001006b66d0 0x0005e03100000007

4、总结

上一篇 下一篇

猜你喜欢

热点阅读