本文由快学吧个人写作,以任何形式转载请表明原文出处
一、资料准备
对应mac的版本是11.1。可根据自己的系统版本挑选可以进行调试的源码。
二、思路
- 上一章知道了非懒加载类是如何从一个镜像中加载到了内存中并且被实现,从而真正的被app使用到。
- 那懒加载的类呢?
三、懒加载类的加载实现
1. 懒加载类和非懒加载类的区别
懒加载类 : 在第一次调用的时候才完成实现和加载。
非懒加载类 : 在程序初始化的时候就进行了加载。
例如 : 在一个类的实现中写入一个+(void)load方法,哪怕+(void)load中什么都不写,这个类也会变成非懒加载类。
2. 懒加载类的实现
1. 创建一个懒加载类
既然懒加载类没有在最开始的程序初始化时进行加载,而是在第一次调用的时候完成加载,那么可以举例 :
- 在objc818.2中创建一个
JDMan类,继承于NSObject。添加一些实例变量、属性和方法,但是不要在JDMan.m中写入+(void)load方法。

- 现在的
JDMan是懒加载的,可以验证。进入到_objc_init(void)--->map_images--->map_images_nolock--->_read_images。找到非懒加载类的实现的for循环。在其中加入如下代码,获取mach-o静态段中保存的所有的非懒加载类 :

执行代码,会获得到mach-o中的非懒加载字段中的所有非懒加载类,在lldb中搜索:JDMan,是找不到的 :

证明JDMan不在可执行文件的非懒加载的表中,所以JDMan是懒加载类。
2. 懒加载类的实现
- 在
main.m文件中实例化一个JDMan的实例 :

-
当
JDMan这个懒加载类调用到alloc这个方法的时候,就是JDMan在被使用了,那么JDMan就需要被实现。 -
JDMan调用alloc方法的本质是objc_magSend,这个之前说方法的本质的时候说过。既然是objc_msgSend,那么就会进行缓存查找或lookUpImpOrForward,JDMan是第一次被使用,所以缓存中必然是不存在alloc方法的缓存的,那么就会找到lookUpImpOrForward。 -
lookUpImpOrForward中是有检查类是否实现的,进入lookUpImpOrForward源码,如下图 :

-
但是,这里有一个问题,
[JDMan alloc]这个是类在调用类方法,那么lookUpImpOrForward中的cls应该是JDMan元类,inst才是JDMan类。 -
先进入
realizeAndInitializeIfNeeded_locked源码 :

- 一步一步的进入
initializeAndLeaveLocked:

- 上面那一步注意,第一个参数传的是
nonmeta,是JDMan类。一步一步的进入realizeClassMaybeSwiftAndUnlock:


3. 懒加载类的加载的结论 :
懒加载类是通过lookUpImpOrForward进入到的
realizeClassWithoutSwift,在上一章已经说过realizeClassWithoutSwift是怎么实现一个类的,不再重复。