首先我们之前知道isa的指针的指向结论是:
1.instance对象的isa指向class对象
2.clsaa对象的isa指向meta-class
3.meta-class对象的isa指向基类的meta-class对象
这篇博客,我们将探究isa指针指向哪里?并且会从代码层的内存地址证明isa指针的指向。
首先instance对象的isa指针指向class对象,我们就可以创建一个instance对象
从上面的截图运行结果可以看出stu的isa的0x001d8001000084e9 和GDStudent的isa的0x00000001000084e8并不一样,跟我们之前说的结果不一致?
其实啊,在arm64之前,也就是iPhone 5s之前其实是一致的,而在arm64以后它需要一个&位运算操作,需要stu的isa与ISA_MASK这个常量位运算一次,也就是
0x001d8001000084e9 & ISA_MASK ,现在我们苹果源码看一下ISA_MASK是什么东西,请看下面的截图
需要这份源码的,请移步下载:苹果源码下载地址
因为我们创建的是mac程序,所以我们复制下面的ISA_MASK位运算操作一次,请看下面的结果:
从红色的结果是不是一摸一样,正好验证了instance对象的isa指针指向class对象!
此时是不是666!😄
2.clsaa对象的isa指向meta-class
细心的同学可能可能发现我们无法输出classStu的isa,就像上面的绿色不是structure,我们点一下class的jump to Definition查看它的源码定义
这样我们看最后那种红色部分,确实是有个isa,此时怎么处理呢?我们可以定义一个一摸一样的结构体来替换一下class就行了如下代码:
struct gd_objc_class {
Class isa;
Class superclass;
};
objc_class前面加gd是防止重名,编译不通过,所以随便写啥都行
现在就可以验证了
这样结果就是一摸一样可以验证了!
这样也是从代码层验证了
1.instance对象的isa指向class对象
2.clsaa对象的isa指向meta-class
3.meta-class对象的isa指向基类的meta-class对象
接下来我将用代码证明
1.对象方法、属性、成员变量、协议信息存放在class对象中
2.类方法存放在meta-class对象中
3.成员变量的具体值是存放在instance对象中