上一篇分析了SpringBoot启动时广播生命周期事件的原理,关键步骤总结:
- 为广播SpringBoot内置生命周期事件做前期准备:
1)首先加载ApplicationListener
监听器实现类;
2)其次加载SPI扩展类EventPublishingRunListener
。 - SpringBoot启动时利用
EventPublishingRunListener
广播生命周期事件,然后ApplicationListener
监听器实现类监听相应的生命周期事件执行一些初始化逻辑的工作。
1.SpringBoot生命周期事件源码分析
SpringBoot的生命周期事件,类结构图:
由上图可以看到事件类之间的关系:
- 最顶级的父类是JDK的事件基类
EventObject
; - 然后Spring的事件基类
ApplicationEvent
继承了JDK的事件基类EventObject
; - 其次SpringBoot的生命周期事件基类
SpringApplicationEvent
继承了Spring的事件基类ApplicationEvent
; - 最后SpringBoot具体的7个生命周期事件类再继承了SpringBoot的生命周期事件基类
SpringApplicationEvent
。
1.1 JDK的事件基类EventObject
EventObject
类是JDK的事件基类,是所有Java事件类的基本,即所有的Java事件类都直接或间接继承于该类,源码如下:
1 | public class EventObject implements java.io.Serializable { |
可以看到EventObject
类只有一个属性source
,这个属性是用来记录最初事件是发生在哪个类。
比如在SpringBoot启动过程中会触发ApplicationStartingEvent
事件,而这个事件最初是在SpringApplication
类中触发的,因此source
就是SpringApplication
对象。
1.2 Spring的事件基类ApplicationEvent
ApplicationEvent
继承了JDK的事件基类EventObject
类,是Spring的事件基类,被所有Spring的具体事件类继承,源码如下:
1 | public abstract class ApplicationEvent extends EventObject { |
可以看到ApplicationEvent
有且仅有一个属性timestamp
,该属性是用来记录事件发生的时间。
1.3 SpringBoot的事件基类SpringApplicationEvent
SpringApplicationEvent
类继承了Spring的事件基类ApplicationEvent
,是所有SpringBoot内置生命周期事件的父类,源码如下:
1 | "serial") ( |
可以看到SpringApplicationEvent
有且仅有一个属性args
,该属性就是SpringBoot启动时的命令行参数即标注@SpringBootApplication
启动类中main
函数的参数。
2.SpringBoot具体的生命周期事件类
接下来分析一下SpringBoot
内置生命周期事件即SpringApplicationEvent
的7个具体子类。
2.1 ApplicationStartingEvent
1 | public class ApplicationStartingEvent extends SpringApplicationEvent { |
SpringBoot开始启动时便会发布ApplicationStartingEvent
事件,其发布时机在环境变量Environment或容器ApplicationContext创建前但在注册ApplicationListener
具体监听器之后,标志标志SpringApplication
开始启动。
2.2 ApplicationEnvironmentPreparedEvent
1 | public class ApplicationEnvironmentPreparedEvent extends SpringApplicationEvent { |
可以看到ApplicationEnvironmentPreparedEvent
事件多了一个environment
属性,作用是利用事件发布订阅机制,相应监听器可以从ApplicationEnvironmentPreparedEvent
事件中取出environment
变量,然后可以为environment
属性增加属性值或读出environment
变量中的值。
例如:
ConfigFileApplicationListener
监听器就是监听了ApplicationEnvironmentPreparedEvent
事件,然后取出ApplicationEnvironmentPreparedEvent
事件的environment
属性,然后再为environment
属性增加application.properties
配置文件中的环境变量值。
当SpringApplication已经开始启动且环境变量Environment
已经创建后,并且为环境变量Environment
配置了命令行和Servlet
等类型的环境变量后,此时会发布ApplicationEnvironmentPreparedEvent
事件。
监听ApplicationEnvironmentPreparedEvent
事件的第一个监听器是ConfigFileApplicationListener
,因为是ConfigFileApplicationListener
监听器还要为环境变量Environment
增加application.properties
配置文件中的环境变量;此后还有一些也是监听ApplicationEnvironmentPreparedEvent
事件的其他监听器监听到此事件时,此时环境变量Environment
几乎已经完全准备好了。
思考: 监听同一事件的监听器们执行监听逻辑时是有顺序的,可以想一下这个排序逻辑是什么时候排序的?还有为什么要这样排序呢?
2.3 ApplicationContextInitializedEvent
1 | public class ApplicationContextInitializedEvent extends SpringApplicationEvent { |
可以看到ApplicationContextInitializedEvent
事件多了个ConfigurableApplicationContext
类型的context
属性,context
属性的作用同样是为了相应监听器可以拿到这个context
属性执行一些逻辑。
ApplicationContextInitializedEvent
事件在ApplicationContext
容器创建后,且为ApplicationContext
容器设置了environment
变量和执行了ApplicationContextInitializers
的初始化方法后但在bean定义加载前触发,标志ApplicationContext已经初始化完毕。
扩展: 可以看到
ApplicationContextInitializedEvent
是在为context
容器配置environment
变量后触发,此时ApplicationContextInitializedEvent
等事件只要有context
容器的话,那么其他需要environment
环境变量的监听器只需要从context
中取出environment
变量即可,从而ApplicationContextInitializedEvent
等事件没必要再配置environment
属性。
2.4 ApplicationPreparedEvent
1 | public class ApplicationPreparedEvent extends SpringApplicationEvent { |
同样可以看到ApplicationPreparedEvent
事件多了个ConfigurableApplicationContext
类型的context
属性,多了context
属性的作用是能让监听该事件的监听器们能拿到context
属性,监听器拿到context
属性一般有如下作用:
- 从事件中取出
context
属性,然后可以增加一些后置处理器,比如ConfigFileApplicationListener
监听器监听到ApplicationPreparedEvent
事件后,然后取出context
变量,通过context
变量增加了PropertySourceOrderingPostProcessor
这个后置处理器; - 通过
context
属性取出beanFactory
容器,然后注册一些bean
,比如LoggingApplicationListener
监听器通过ApplicationPreparedEvent
事件的context
属性取出beanFactory
容器,然后注册了springBootLoggingSystem
这个单例bean
; - 通过
context
属性取出Environment
环境变量,然后就可以操作环境变量,比如PropertiesMigrationListener
。
ApplicationPreparedEvent
事件在ApplicationContext
容器已经完全准备好时但在容器刷新前触发,在这个阶段bean
定义已经加载完毕还有environment
已经准备好可以用了。
2.5 ApplicationStartedEvent
1 | public class ApplicationStartedEvent extends SpringApplicationEvent { |
ApplicationStartedEvent
事件将在容器刷新后但ApplicationRunner
和CommandLineRunner
的run
方法执行前触发,标志Spring
容器已经刷新,此时容器已经准备完毕了。
扩展: 这里提到了
ApplicationRunner
和CommandLineRunner
接口有啥作用呢?一般会在Spring
容器刷新完毕后,此时可能有一些系统参数等静态数据需要加载,此时就可以实现了ApplicationRunner
或CommandLineRunner
接口来实现静态数据的加载。
2.6 ApplicationReadyEvent
1 | public class ApplicationReadyEvent extends SpringApplicationEvent { |
ApplicationReadyEvent
事件在调用完ApplicationRunner
和CommandLineRunner
的run
方法后触发,此时标志SpringApplication
已经正在运行。
2.7 ApplicationFailedEvent
1 | public class ApplicationFailedEvent extends SpringApplicationEvent { |
可以看到ApplicationFailedEvent
事件除了多了一个context
属性外,还多了一个Throwable
类型的exception
属性用来记录SpringBoot启动失败时的异常。
ApplicationFailedEvent
事件在SpringBoot启动失败时触发,标志SpringBoot启动失败。
总结
SpringBoot启动过程中要触发的各种生命周期事件
发布顺序 | 时间 | 用途 |
---|---|---|
1 | ApplicationStartingEvent | 在SpringApplication启动时,在环境变量Environment或者容器ApplicationContext创建前触发,标志SpringApplication开始启动。 |
2 | ApplicationEnvironmentPreparedEvent | 当SpringApplication已经开始启动且环境变量Environment已经准备好时触发,标志环境变量已经准备好。 |
3 | ApplicationContextInitializedEvent | ApplicationContextInitializers的初始化方法已经被调用,即从spring.factories中加载的initializers已经执行ApplicationContext初始化逻辑但在bean定义加载前触发,标志ApplicationContext已经初始化完毕。 |
4 | ApplicationPreparedEvent | 在Spring容器刷新refresh前触发 |
5 | ApplicationStartedEvent | 在spring容器刷新后触发,但在调用ApplicationRunner和CommandLineRunner的run方法调用前触发,标志spring容器已经刷新,此时所有的bean实例等都已经加载了。 |
6 | ApplicationReadyEvent | 只要SpringApplication可以接收服务请求时即调用完ApplicationRunner和CommandLineRunner的run方法后触发,此时标志SpringApplication已经正在运行,即启动成功。 |
7 | ApplicationFailedEvent | 若SpringApplication未能成功启动时则会catch住异常发布ApplicationFailedEvent事件,标志ApplicationFailedEvent启动失败。 |
评论加载中