1 引言
在SpringBoot启动过程中,每个不同的启动阶段会分别广播不同的内置生命周期事件,然后相应的监听器会监听这些事件来执行一些初始化逻辑工作比如ConfigFileApplicationListener
会监听onApplicationEnvironmentPreparedEvent
事件来加载配置文件application.properties
的环境变量等。
因此本篇内容将来分析下SpringBoot的事件监听机制的源码。
2 SpringBoot广播内置生命周期事件流程分析
为了探究SpringBoot广播内置生命周期事件流程,我们再来回顾一下SpringBoot的启动流程代码:
1 | // SpringApplication.java |
可以看到SpringBoot在启动过程中首先会先新建一个SpringApplicationRunListeners
对象用于发射SpringBoot启动过程中的各种生命周期事件,比如发射ApplicationStartingEvent
,ApplicationEnvironmentPreparedEvent
和ApplicationContextInitializedEvent
等事件,然后相应的监听器会执行一些SpringBoot启动过程中的初始化逻辑。那么,监听这些SpringBoot的生命周期事件的监听器们是何时被加载实例化的呢?还记得上篇文章在分析SpringApplication
的构建过程吗?没错,这些执行初始化逻辑的监听器们正是在SpringApplication
的构建过程中根据ApplicationListener
接口去spring.factories
配置文件中加载并实例化的。
2.1 为广播SpringBoot内置生命周期事件做前期准备
2.1.1 加载ApplicationListener监听器实现类
我们再来回顾下SpringApplication对象是如何构建的? SpringBoot源码(八)一文中讲到在构建SpringApplication
对象时的setListeners((Collection) getSpringFactoriesInstances(ApplicationListener.class));
这句代码。
这句代码做的事情就是从spring.factories
中加载出ApplicationListener
事件监听接口的SPI扩展实现类然后添加到SpringApplication
对象的listeners
集合中,用于后续监听SpringBoot启动过程中的事件,来执行一些初始化逻辑工作。
SpringBoot启动时的具体监听器们都实现了ApplicationListener
接口,其在spring.factories
部分配置如下:
不过在调试时,会从所有的spring.factories配置文件中加载监听器,最终加载了10个监听器。如下图:
"2.1.2 加载SPI扩展类EventPublishingRunListener
前面讲到,在SpringBoot的启动过程中首先会先新建一个SpringApplicationRunListeners
对象用于发射SpringBoot启动过程中的生命周期事件,即我们现在来看下SpringApplicationRunListeners listeners = getRunListeners(args);
这句代码:
1 | // SpringApplication.java |
我们将重点放到getSpringFactoriesInstances( SpringApplicationRunListener.class, types, this, args)
这句代码,getSpringFactoriesInstances
这个方法我们已经很熟悉,在上一篇分析SpringBoot的SPI机制时已经详细分析过这个方法。可以看到SpringBoot此时又是根据SpringApplicationRunListener
这个SPI接口去spring.factories
中加载相应的SPI扩展实现类,我们直接去spring.factories
中看看SpringApplicationRunListener
有哪些SPI实现类:
由上图可以看到,SpringApplicationRunListener
只有EventPublishingRunListener
这个SPI实现类 EventPublishingRunListener
这个哥们在SpringBoot的启动过程中尤其重要,由其在SpringBoot启动过程的不同阶段发射不同的SpringBoot的生命周期事件,即SpringApplicationRunListeners
对象没有承担广播事件的职责,而最终是委托EventPublishingRunListener
这个哥们来广播事件的。
因为从spring.factories
中加载EventPublishingRunListener
类后还会实例化该类,那么我们再跟进EventPublishingRunListener
的源码,看看其是如何承担发射SpringBoot生命周期事件这一职责的?
1 | // EventPublishingRunListener.java |
可以看到EventPublishingRunListener
类实现了SpringApplicationRunListener
接口,SpringApplicationRunListener
接口定义了SpringBoot启动时发射生命周期事件的接口方法,而EventPublishingRunListener
类正是通过实现SpringApplicationRunListener
接口的starting
,environmentPrepared
和contextPrepared
等方法来广播SpringBoot不同的生命周期事件,我们直接看下SpringApplicationRunListener
接口源码好了:
1 | // SpringApplicationRunListener.java |
我们再接着分析EventPublishingRunListener
这个类,可以看到其有一个重要的成员属性initialMulticaster
,该成员属性是SimpleApplicationEventMulticaster
类对象,该类正是承担了广播SpringBoot启动时生命周期事件的职责,即EventPublishingRunListener
对象没有承担广播事件的职责,而最终是委托SimpleApplicationEventMulticaster
这个哥们来广播事件的。 从EventPublishingRunListener
的源码中也可以看到在starting
,environmentPrepared
和contextPrepared
等方法中也正是通过调用SimpleApplicationEventMulticaster
类对象的multicastEvent
方法来广播事件的。
思考 SpringBoot启动过程中发射事件时事件广播者是层层委托职责的,起初由
SpringApplicationRunListeners
对象承担,然后SpringApplicationRunListeners
对象将广播事件职责委托给EventPublishingRunListener
对象,最终EventPublishingRunListener
对象将广播事件的职责委托给SimpleApplicationEventMulticaster
对象。为什么要层层委托这么做呢? 这个值得大家思考。
前面讲到从spring.factories
中加载出EventPublishingRunListener
类后会实例化,而实例化必然会通过EventPublishingRunListener
的构造函数来进行实例化,因此我们接下来分析下EventPublishingRunListener
的构造函数源码:
1 | // EventPublishingRunListener.java |
可以看到在EventPublishingRunListener
的构造函数中有一个for
循环会遍历之前从spring.factories
中加载的监听器们,然后添加到集合中缓存起来,用于以后广播各种事件时直接从这个集合中取出来即可,而不用再去spring.factories
中加载,提高效率。
2.2 广播SpringBoot的内置生命周期事件
从spring.factories
配置文件中加载并实例化EventPublishingRunListener
对象后,那么在在SpringBoot的启动过程中会发射一系列SpringBoot内置的生命周期事件,我们再来回顾下SpringBoot启动过程中的源码:
1 | // SpringApplication.java |
可以看到在SpringBoot的启动过程中总共会发射7种不同类型的生命周期事件,来标志SpringBoot的不同启动阶段,同时,这些生命周期事件的监听器们也会执行一些启动过程中的初始化逻辑,关于这些监听器的初始化逻辑将在下一篇内容中会分析。以下是SpringBoot启动过程中要发射的事件类型,其中ApplicationFailedEvent
在SpringBoot启动过程中遇到异常才会发射:
ApplicationStartingEvent
ApplicationEnvironmentPreparedEvent
ApplicationContextInitializedEvent
ApplicationPreparedEvent
ApplicationStartedEvent
ApplicationFailedEvent
ApplicationReadyEvent
我们以listeners.starting();
这句代码为例,看看EventPublishingRunListener
对象发射事件的源码:
1 | // SpringApplicationRunListeners.java |
继续跟进listener.starting();
的源码:
1 | EventPublishingRunListener.java |
可以看到,EventPublishingRunListener
对象将发布ApplicationStartingEvent
这件事情委托给了SimpleApplicationEventMulticaster
对象initialMulticaster
, ,而initialMulticaster
对象最终会调用其multicastEvent
方法来发射ApplicationStartingEvent
事件。关于SimpleApplicationEventMulticaster
类如何广播事件,笔者已经在Spring是如何实现事件监听机制的? Spring源码(二)这篇文章已经详细分析,这里不再赘述。
关于SpringBoot启动过程中发射其他生命周期事件的源码这里不再分析
3 SpringBoot的内置生命周期事件总结
好了,前面已经分析了SpringBoot启动过程中要发射的各种生命周期事件,下面列一个表格总结下:
"4 小结
SpringBoot启动过程中广播生命周期事件的源码分析就到此结束了,下一篇会继续介绍监听这些生命周期事件的监听器们。我们再回顾本篇内容总结下关键点:
SpringBoot启动过程中会发射7种类型的生命周期事件,标志不同的启动阶段,然后相应的监听器会监听这些事件来执行一些初始化逻辑工作。