OC底层原理06-isa流程图
2020-09-12 本文已影响0人
Gomu_iOS
一、回顾
上一期OC底层原理05-isa结构分析中,我们在GomuPerson
的实例对象person
的地址中,拿到第一位地址中的isa
,进行逆向推导isa
&mask
得到GomuPerson
类
//: 第一步,拿到`person`的内存地址
(lldb) x/4gx person
//: 或者
(lldb) p person
(GomuPerson *) $31 = 0x0000000100632a50
(lldb) x/4gx 0x0000000100632a50
//: 两种方式都能拿到`person`的地址
//: 下面的`0x001d800100002465 `就是我们需要的`isa`
0x100632a50: 0x001d800100002465 0x0000000000000000
0x100632a60: 0x0000000000000000 0x0000000000000000
//: 用`isa` & `mask`
(lldb) p/x 0x001d800100002465 & 0x00007ffffffffff8ULL
//: 得到了`GomuPerson `的16进制地址
(unsigned long long) $29 = 0x0000000100002460
//: 得到`GomuPerson `
(lldb) po 0x0000000100002460
GomuPerson
二、isa流程探索
既然我们用x/4gx
打印person
对象的地址,找到isa
,从而拿到GomuPerson
,那我们用x/4gx
打印GomuPerson
的地址,它会有isa
吗?而它的isa
又会拿到什么?
2.1 类的isa
的指向
//: 拿到`GomuPerson `的内存地址
(lldb) x/4gx 0x0000000100002460
0x100002460: 0x0000000100002438 0x0000000100334140
0x100002470: 0x0000000100632a80 0x0002803400000003
//: `isa` & `mask`
(lldb) p/x 0x0000000100002438 & 0x00007ffffffffff8ULL
(unsigned long long) $32 = 0x0000000100002438
//: 又得到了`GomuPerson`
(lldb) po 0x0000000100002438
GomuPerson
-
对象的isa
->类
(person -> GomuPerson),类的isa
->?
(GomuPerson的isa -> GomuPerson ) -
对象
和类
都有isa
,说明类也是对象
-
OC底层原理01-alloc流程探索中已经知道了,
类对象
在内存中只会存一份
,但是这里po 0x0000000100002438
和po 0x0000000100002460
都能拿到GomuPerson
,注意这里第二次打印的GomuPerson
是元类
-
元类的定义和创建
是由编译器自动完成
- 得出
isa
的流程对象的isa
->类的isa
->元类
2.2 元类的isa
的指向
//: 拿到`GomuPerson `元类的内存地址
(lldb) x/4gx 0x0000000100002438
0x100002438: 0x00000001003340f0 0x00000001003340f0
0x100002448: 0x00000001010045e0 0x0005e03500000007
//: `isa` & `mask`
(lldb) p/x 0x00000001003340f0 & 0x00007ffffffffff8ULL
(unsigned long long) $34 = 0x00000001003340f0
//: 得到`NSObject `
(lldb) po 0x00000001003340f0
NSObject
-
元类的isa
指向了NSObject
- 得出
isa
的流程对象的isa
->类的isa
->元类的isa
->NSObject
2.3 NSObject的isa
的指向
//: 拿到`NSObject `的内存地址
(lldb) x/4gx 0x00000001003340f0
0x1003340f0: 0x00000001003340f0 0x0000000100334140
0x100334100: 0x0000000101059460 0x0005e03100000007
//: `isa` & `mask`
(lldb) p/x 0x00000001003340f0 & 0x00007ffffffffff8ULL
(unsigned long long) $36 = 0x00000001003340f0
//: 又得到`NSObject `
(lldb) po 0x00000001003340f0
NSObject
-
NSObject
又指向了NSObject
- 两个
NSObject
的地址都是0x00000001003340f0
,说明这里的NSObject的isa
会指向自己 - 得出
isa
的流程对象的isa
->类的isa
->元类的isa
->NSObject的isa
->NSObject
2.4 这里的NSObject
和系统初始化的NSObject
的区别
//: 获取系统初始化的`NSObject `的地址
(lldb) p/x NSObject.class
(Class) $43 = 0x0000000100334140 NSObject
//: 拿到系统初始化的`NSObject `的`isa`
(lldb) x/4gx 0x0000000100334140
0x100334140: 0x00000001003340f0 0x0000000000000000
0x100334150: 0x0000000100632d90 0x0001801000000003
- 系统初始化的
NSObject
的isa
指向了0x00000001003340f0
,和我们上面获取到的NSObjcet
的地址相同 -
GomuPerson
元类的isa指向的NSObject
,不是系统的NSObjcet
,而是系统的NSObjcet
的元类
,即根元类
- 得出
isa
的流程对象的isa
->类的isa
->元类的isa
->根元类的isa
->根元类
- 万物皆对象,皆有
isa
2.5 打印对象地址验证流程
// 实例对象
GomuPerson *person = [GomuPerson alloc];
// 类
Class class = object_getClass(person);
// 元类
Class metaClass = object_getClass(class);
// 根元类
Class rootMetaClass = object_getClass(metaClass);
// 根根元类
Class rootRootMetaClass = object_getClass(rootMetaClass);
NSLog(@"\n%p 实例对象\n%p 类\n%p 元类\n%p 根元类\n%p 根根元类",person,class,metaClass,rootMetaClass,rootRootMetaClass);
//: 输出
0x10066fe20 实例对象
0x100002470 类
0x100002448 元类
0x1003340f0 根元类
0x1003340f0 根根元类
2.6 isa
流程图
9491599893819_.pic_hd.jpg
三、继承流程图
- 实例对象没有继承关系,继承关系来自于类
-
GomuBoy
继承于GomuPerson
-
person1
,person2
是GomuPerson
的实例对象 -
boy1
,boy2
是GomuBoy
的实例对象 -
boy1
,boy2
与person1
,person2
没有继承关系
-
- 类和元类才有继承关系
-
NSObject
继承于nil
-
根元类
继承于NSObject
image.png