基于springboot 2.6.6版本的debug记录
在主配置类的main方法调试进入后,程序是先new一个SpringApplication对象,然后再调用对象的run方法,那么可以分解为new 和run两部分来详细说一下
1创建SpringApplication对象
在它的有参构造函数中执行几个步骤如下:
1首先校验主配置类是否为null,如果为null则报非法参数异常
2接着调用WebApplicationType.deduceFromClasspath()方法来设置当前应用类型
3接着调用getSpringFactoriesInstances(xxx.class)方法
去meta-info包下的spring.factories文件中找
引导注册初始化(org.springframework.boot.BootstrapRegistryInitializer)的配置信息,并保存到SpringApplication对象的bootstrapRegistryInitializers属性中
4接着又调用getSpringFactoriesInstances(xxx.class)
方法在spring.factories文件中找
应用上下文初始化(org.springframework.boot.ApplicationContextInitializer)的配置信息,并保存到SpringApplication对象的initializers属性中
5接着又调用getSpringFactoriesInstances(xxx.class)
方法在spring.factories文件中找应用监听器
(org.springframework.context.ApplicationListener)的配置信息并保存到SpringApplication对象的listeners属性中,以上三个属性都是list集合
6最后调用deduceMainApplicationClass()方法来设置主程序类
2执行SpringApplication对象的run()方法
1记录应用的启动时间
2执行createBootstrapContext()方法创建引导上下文对象,在这个方法中,是去遍历了SpringApplication对象的bootstrapRegistryInitializer属性中的对象执行了initialize方法来完成引导初始化的环境设置
3让当前应用进入Headless模式(自力更生模式)
4获取所有runListeners运行监听器,还是去spring.factories文件中找
org.springframework.boot.SpringApplicationRunListener的配置信息
5遍历SpringApplicationRunListeners对象的starting方法来通知项目正在启动中
6保存命令行参数
7调用prepareEnvironment()准备环境信息
该方法中对所有的配置文件的属性值还有一些环境变量信息命令行参数等做配置处理
最后调用environmentPrepared通知所有的监听器当前环境已经准备完成
8调用createApplicationContext()创建IOC容器,会根据项目类型(servlet)创建ico容器。
9调用prepareContext准备ApplicationContext IOC容器的基本信息
保存环境信息
IOC容器的后置处理流程
应用初始化器,就是遍历所有ApplicationContextInitializer,调用initialize来对容器初始化的扩展功能
调用contextPrepared方法通知所有监听器,IOC容器的上下文已经准备好了
调用contextLoaded通知所有监听器IOC容器上下文已经加载
10调用refreshContext刷新IOC容器,就是spring经典的整个初始化过程,创建容器中的所有组件
11调用afterRefresh,容器刷新完成后的一些工作处理,只是个空方法
12调用started通知所有的监听器项目已启动
13调用callRunners()方法去执行ApplicationRunner和CommandLineRunner的run方法,比如项目启动的时候去读取配置文件、或者把数据库的数据加载到缓存中的时候就可以帮助我们实现这些需求。怎么用?好像是写一个ApplicationRunner或CommandLineRunner接口的实现类,并且实现类添加到IOC容器中,调用callRunners就会调用到实现类的run方法了
14记录整个应用所花费时间,没有任何异常最后会调用所有监听器的ready方法,在上面这些过程中如果有异常,还会调用所有监听器的failed方法