本文除了讲述isa
指针, 也讲了superclass
指针
要讲清楚isa
指针和superclass
指针, 事实上只要搞清楚一张简单的图就好了
![](https://img.haomeiwen.com/i2868984/ffacd39a39831170.png)
原图链接
简单总结一下(这也是很经典的一道面试题):
instance
对象的isa
指针是指向它的class
对象的
class
对象的isa
指针是指向它的meta-class
对象的
meta-class
对象的isa
指针是指向基类的meta-class
对象的
基类的meta-class
对象的isa
指针是指向它自己的
class
对象的superclass
指针是指向父类的class
对象的
父类的class
对象的superclass
指针是指向它的父类的class
对象的
一直指向基类的class
对象
基类的class
对象的superclass
指针是nil
meta-class
对象的superclass
指针是指向父类的meta-class
对象的
父类的meta-class
对象的superclass
指针是指向它的父类的meta-class
对象的
一直指向基类的meta-class
对象
基类的meta-class
对象的superclass
指针是指向基类的class
对象的(这一点要尤其注意)
几个要注意的点:
在64
位操作系统中, isa
指针并不是直接指向的
举个例子:
比如简单理解, instance
对象的isa
指针是指向class
对象的. 但其实并不是直接指向的, 而是要与&
上一个ISA_MASK
比如这段代码
![](https://img.haomeiwen.com/i2868984/06c781c191190560.png)
object1
的地址就是isa
的地址, 那么isa
指针指向的那片内存空间的值是多少呢?
![](https://img.haomeiwen.com/i2868984/17f66fce5ee658e4.png)
是0x01000001d771ef31
![](https://img.haomeiwen.com/i2868984/786cc05f994c6c6b.png)
而classObject1
的地址是0x00000001d771ef30
, 这中间就通过了一次位运算
# if __arm64__
# define ISA_MASK 0xfffffffffffffff8ULL
# elif __x86_64__
# define ISA_MASK 0x00007ffffffffff8ULL
简单来说, 我现在用的是x86
架构, ISA_MASK
的值就为0x00007ffffffffff8ULL
, 所以, 用isa
指针的0x01000001d771ef31
值, 与上
0x00007ffffffffff8ULL
, 刚好就是0x00000001d771ef30
, 就是classObject1
这个类对象的值
![](https://img.haomeiwen.com/i2868984/d79a32e287952d70.png)
但是有些奇怪的是类对象的isa
现在是直接指向元类对象了
![](https://img.haomeiwen.com/i2868984/a5ea03898d43a32c.png)
这里也可以看出来, 基类的元类对象的isa
指针, 是指向自己的
![](https://img.haomeiwen.com/i2868984/374026bbb35bc674.png)
需要注意的一点是: superclass
是直接指向的
![](https://img.haomeiwen.com/i2868984/275c1cc7eb372b60.png)
感觉可以总结出的一点是:
instance
对象的isa
指针做了优化, 并不是一个isa
指针直接指向class
对象, 而是进行了一次位运算. 这么做的目的是什么, 我放在
isa指针(二)这篇文章中说了.
不管是isa
指针还是superclass
指针, 都没有经过位运算, 而是直接指向了元类对象(isa
指针), 或者父类的类对象(superclass
指针)
OC
调用方法的本质是给对象发消息(消息机制)
![](https://img.haomeiwen.com/i2868984/2135339ab08ce493.png)
比如,
student
调用studentInstanceMethod
的本质就是
((void (*)(id, SEL))(void *)objc_msgSend)((id)student, sel_registerName("studentInstanceMethod"));
简化之后就是
objc_msgSend(student, sel_registerName("studentInstanceMethod")
![](https://img.haomeiwen.com/i2868984/c257ef86e9737496.png)
可以很清楚的看到, 调用实例方法就是给实例方法发消息, 调用类方法就是给类对象发消息
基本的原理就是:
instance
对象通过isa
指针找到对应的class
对象, 在class
对象的方法列表中找到对应的实例方法, 然后调用objc_msgSend
class
对象通过isa
指针找到对应的meta-class
对象, 在meta-class
对象的方法列表中找到对应的类方法, 然后调用objc_msgSend
如果找不到怎么办?
就会通过superclass
指针找到父类, 在父类的方法列表中, 找对应的方法
![](https://img.haomeiwen.com/i2868984/43550fb364f1a97b.png)
如果一直找到基类都找不到, 就会报错
![](https://img.haomeiwen.com/i2868984/5a8d21049699ef01.png)
unrecognized selector sent to instance 0x600000b8c040'
有一个问题需要注意的是:
![](https://img.haomeiwen.com/i2868984/2b71b10e43e602f3.png)
上图中红色框标记的, 假设基类的
meta-class
对象中, 本应该存储某一个类方法的, 但是找不到这个类方法, 那么就根据superclass
指向的父类的方法列表去找, 但是这个superclass
指向的是基类的class
对象, 但class
对象中是存储的对象方法, 那么, 能找到吗?
答案是, 如果是同名方法, 是可以的. OC
的消息机制, 本质上是给对象发消息, 是不区分类方法和实例方法的. OC
只是在面向对象层名区分了类方法和实例方法, 但是本质上, 是不区分的, 只是根据方法名发消息, 那么, 如何验证呢?
![](https://img.haomeiwen.com/i2868984/5a4283364c6ea1a7.png)
我写了一个实例方法
test
, 但是暴露的接口是类方法:![](https://img.haomeiwen.com/i2868984/338cfe46a5ad699e.png)
调用的时候依然可以调到:
![](https://img.haomeiwen.com/i2868984/dcb4074582af5fb5.png)
总结一道面试题:
OC
的类信息存放在哪里?
- 对象方法, 属性, 成员变量, 协议信息, 存放在
class
对象中 - 类方法, 存放在
meta-class
对象中 - 成员变量的具体值, 存放在
instance
对象
网友评论