回顾
在之前的几篇博客里面,我们知道了对象的本质是结构体(iOS底层探索之对象的本质和类的关联特性initIsa(上)),也了解了结构体的内存对齐(结构体底层探索),也分析了下alloc的底层的源码(alloc底层探索),也知道了实例化一个对象,底层是通过isa和类进行关联的。但一直都没有探索下类,那么接下来,我们就去底层看看类,揭开类的神秘面纱。

类的猜想
首先看看下面👇这个,我们实例化一个对象,在断点处进行调试

通过打印对象stu的地址,得到ISA,然后和掩码0x00007ffffffffff8作&运算,可以得到类
(lldb) x/4gx stu
0x10072a850: 0x011d800100008339 0x0000000000000000
0x10072a860: 0x0000000000000000 0x0000000000000000
(lldb) p/x 0x011d800100008339 & 0x00007ffffffffff8
(long) $1 = 0x0000000100008338
(lldb) po 0x0000000100008338
JPStudent
那么我们再作一个猜想,类是不是也有一个ISA,那么同步上面的操作看看
(lldb) x/4gx 0x0000000100008338
0x100008338: 0x0000000100008310 0x00000001000082c0
0x100008348: 0x00000001006ac3e0 0x000180200000000f
(lldb) p/x 0x0000000100008310 & 0x00007ffffffffff8
(long) $3 = 0x0000000100008310
(lldb) po 0x0000000100008310
JPStudent
我的天哪!

打印出来的是一模模一样样,都打印了JPStudent,地址完全不一样啊!一个是0x0000000100008338,一个是0x0000000100008310。那么类和对象一样,会在内存中无限开辟,不只存在一个类吗?那么我们去验证一下!
//MARK: - 分析类对象内存存在个数
void jpTestClassNum(void){
Class class1 = [JPStudent class];
Class class2 = [JPStudent alloc].class;
Class class3 = object_getClass([JPStudent alloc]);
Class class4 = [JPStudent alloc].class;
NSLog(@"\n%p-\n%p-\n%p-\n%p",class1,class2,class3,class4);
}
打印结果
0x100008338-
0x100008338-
0x100008338-
0x100008338
从打印的结果来看,说明类只有一个,类在内存中有且只有一个。
那么0x0000000100008310,显然不是类,那么它是谁呢?是NSObject吗?
元类
(lldb) p/x NSObject.class
(Class) $5 = 0x000000010036a140 NSObject
那么不是NSObject,是什么呢?它是一个新的东西,是什么东东呢?我们通过烂苹果(MachoOView)可以查看

在编译的可以执行文件中,除了我们熟悉的 _OBJC_CLASS_$_JPStudent,还多了一个 _OBJC_METACLASS_$_JPStudent,但是我工程代码里面并没有去创建这个啊?在结构体的内存对齐(结构体底层探索)里面我们知道了,对象的结构里面有个ISA,指向类,那么类的ISA指向谁呢?没错,就是指向元类(META),
元类META是由系统生成和编译的
根元类
那么元类的ISA指向哪里呢?
(lldb) x/4gx 0x0000000100008310
0x100008310: 0x000000010036a0f0 0x00000001000082e8
0x100008320: 0x000000010072adf0 0x0001e03100000007
(lldb) p/x 0x000000010036a0f0 & 0x00007ffffffffff8
(long) $6 = 0x000000010036a0f0
(lldb) po 0x000000010036a0f0
NSObject
从打印结果来看元类的ISA指向了根元类(NSObject),那么根元类的ISA指向哪里呢?

从lldb调试,可以看到根元类指向了自己
对象 isa -> 类 isa -> 元类 isa -> 根元类 isa
根元类 isa -> 根元类 isa ->
来个图呗!这不好理解啊!
好,那么我就来个图,让大家好理解!
isa 走位

从图中可以分析得出每个对象都有元类和根元类,根元类指向自己
代码测试验证
// NSObject实例对象
NSObject *object1 = [NSObject alloc];
// NSObject类
Class class1 = object_getClass(object1);
// NSObject元类
Class metaClass = object_getClass(class1);
// NSObject根元类
Class rootMetaClass = object_getClass(metaClass);
// NSObject根根元类
Class rootRootMetaClass = object_getClass(rootMetaClass);
NSLog(@"\n%p 实例对象\n%p 类\n%p 元类\n%p 根元类\n%p 根根元类",object1,class1,metaClass,rootMetaClass,rootRootMetaClass);
控制台打印
0x10060d620 实例对象
0x10036a140 类
0x10036a0f0 元类
0x10036a0f0 根元类
0x10036a0f0 根根元类
元类继承链
类有继承关系,那么元类也有继承关系
// JPPerson元类
Class pMetaClass = object_getClass(JPPerson.class);
Class psuperClass = class_getSuperclass(pMetaClass);
NSLog(@"%@ - %p",psuperClass,psuperClass);
// JPStudent -> JPPerson -> NSObject
// 元类也有一条继承链
Class sMetaClass = object_getClass(JPStudent.class);
Class ssuperClass = class_getSuperclass(tMetaClass);
NSLog(@"%@ - %p",ssuperClass,ssuperClass);
// NSObject 根类特殊情况
Class nsuperClass = class_getSuperclass(NSObject.class);
NSLog(@"%@ - %p",nsuperClass,nsuperClass);
// 根元类 -> NSObject
Class rnsuperClass = class_getSuperclass(metaClass);
NSLog(@"%@ - %p",rnsuperClass,rnsuperClass);
打印结果
NSObject - 0x10036a0f0
JPPerson - 0x100008300
(null) - 0x0
NSObject - 0x10036a140
从以上代码和运行结果来看
JPPerson的元类是NSObjectNSObject的父类是null,没有爸爸😂NSObject根元类的父类是0x10036a140,这和上面的NSObject的类的地址0x10036a140,是一模模一样样,NSObject根元类又指向了NSObject这个类。
万物皆来自于
NSObject,NSObject指向了null,这和太极里面的,无中生有有点像!
来个图吧!上面的代码看着有点懵啊!

小结
在OC里面,对象主要分为三种,
instance对象(实例对象)class对象(类对象) ,通过object_getClass(传入类)方法获取meta-class对象(元类对象),通过object_getClass(传入元类)方法获取
最后补上一个更详细的图,苹果官方图

类的结构
打开苹果的开源的源码,可以看到类的底层结构

- 默认有个
ISA - 指向父类的指针
superclass cachebits
这个bits有个class_rw_t *,class_rw_t里面有一些方法列表,属性列表,成员变量等等信息,我们看看源码

那么这些我们熟悉的类的信息,该怎么获取,怎么查看呢?
请看下个博客分析。。。。
iOS底层探索之类的结构(中)
🌹请收藏+关注,评论 + 转发,以免你下次找不到我,哈哈😁🌹
🌹欢迎大家留言交流,批评指正,互相学习😁,提升自我🌹

本文详细探讨了Objective-C中对象、类和元类的内部结构。通过LLDB调试,揭示了对象的ISA指针如何指向类,类的ISA指针如何指向元类,以及元类的ISA如何形成一个继承链,最终指向根元类NSObject。作者通过代码示例展示了类对象、元类对象和根元类对象的获取,并分析了类的内存存在个数,验证了类在内存中只有一个。此外,还讨论了元类的继承关系。整个分析过程帮助读者理解Objective-C中的类层次结构和元类概念。
1万+

被折叠的 条评论
为什么被折叠?



