本文由快学吧个人写作,以任何方式转载请表明原文出处

一、材料准备

objc4-818.2

对应mac的版别是11.1。可根据自己的系统版别挑选能够进行调试的源码

二、思路

  1. 上一章知道了非懒加载类是如何从一个镜像中加载到了内存中并且被完成,从而真实的被app运用到。
  2. 那懒加载的类呢?

三、懒加载类的加载完成

1. 懒加载类和非懒加载类的差异

懒加载类 : 在第一次调用的时分才完结完成和加载。

非懒加载类 : 在程序初始化的时分就进行了加载。

例如 : 在一个类的完成中写入一个+(void)load办法,哪怕+(void)load中什么都不写,这个类也会变成非懒加载类。

2. 懒加载类的完成

1. 创立一个懒加载类

既然懒加载类没有在最开始的程序初始化时进行加载,而是在第一次调用的时分完结加载,那么能够举例 :

  1. 在objc818.2中创立一个JDMan类,承继于NSObject。增加一些实例变量、属性和办法,可是不要在JDMan.m中写入+(void)load办法。

二十一、app加载流程(五)类的实现和加载(下)

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

二十一、app加载流程(五)类的实现和加载(下)

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

二十一、app加载流程(五)类的实现和加载(下)

证明JDMan不在可执行文件的非懒加载的表中,所以JDMan是懒加载类。

2. 懒加载类的完成

  1. main.m文件中实例化一个JDMan的实例 :

二十一、app加载流程(五)类的实现和加载(下)

  1. JDMan这个懒加载类调用到alloc这个办法的时分,就是JDMan在被运用了,那么JDMan就需要被完成。

  2. JDMan调用alloc办法的实质是objc_magSend,这个之前说办法的实质的时分说过。既然是objc_msgSend,那么就会进行缓存查找或lookUpImpOrForwardJDMan是第一次被运用,所以缓存中必然是不存在alloc办法的缓存的,那么就会找到lookUpImpOrForward

  3. lookUpImpOrForward中是有检查类是否完成的,进入lookUpImpOrForward源码,如下图 :

二十一、app加载流程(五)类的实现和加载(下)

  1. 可是,这里有一个问题,[JDMan alloc]这个是类在调用类办法,那么lookUpImpOrForward中的cls应该是JDMan元类,inst才是JDMan类。

  2. 先进入realizeAndInitializeIfNeeded_locked源码 :

二十一、app加载流程(五)类的实现和加载(下)

  1. 一步一步的进入initializeAndLeaveLocked :

二十一、app加载流程(五)类的实现和加载(下)

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

二十一、app加载流程(五)类的实现和加载(下)

二十一、app加载流程(五)类的实现和加载(下)

3. 懒加载类的加载的定论 :

懒加载类是通过lookUpImpOrForward进入到的realizeClassWithoutSwift,在上一章已经说过realizeClassWithoutSwift是怎样完成一个类的,不再重复。