目录
[扫描 org.springframework.context.ApplicationListener 指定的类](#扫描 org.springframework.context.ApplicationListener 指定的类)
[spring boot 中的监听](#spring boot 中的监听)
[spring boot autoconfigure 中的监听](#spring boot autoconfigure 中的监听)
[spring boot context 中的监听](#spring boot context 中的监听)
[spring boot 中的监听](#spring boot 中的监听)
[spring boot context 中的监听](#spring boot context 中的监听)
[监听加载到 SpringApplicationRunListeners 中](#监听加载到 SpringApplicationRunListeners 中)
[调用 SpringApplicationRunListeners 的 environmentPrepared() 进行监听调用](#调用 SpringApplicationRunListeners 的 environmentPrepared() 进行监听调用)
在前面的文章基础上
https://blog.csdn.net/zlpzlpzyd/article/details/136065211
spring 相关的项目中的代码一直在变,下面以 spring boot 2.3.12.RELEASE 以及对应的 spring cloud Hoxton.SR12 进行说明。
spring boot 在启动时会通过 SpringFactoriesLoader 加载 classpath 下 META-INF/spring.factories 中的对应的类。
扫描 org.springframework.context.ApplicationListener 指定的类
其中监听对应的接口为 ApplicationListener,对应的监听是其实现类实现对应的功能。
data:image/s3,"s3://crabby-images/56546/56546c000c08bcd2bdd6cc343e339026143ac6dd" alt=""
data:image/s3,"s3://crabby-images/111dc/111dcaa2373cc661b3995ef1a31b40c5579acf1b" alt=""
内置的监听
spring boot 中的监听
data:image/s3,"s3://crabby-images/556f4/556f459b3485504f9c14a0cf132922da4a72a849" alt=""
org.springframework.context.ApplicationListener=\
org.springframework.boot.ClearCachesApplicationListener,\
org.springframework.boot.builder.ParentContextCloserApplicationListener,\
org.springframework.boot.cloud.CloudFoundryVcapEnvironmentPostProcessor,\
org.springframework.boot.context.FileEncodingApplicationListener,\
org.springframework.boot.context.config.AnsiOutputApplicationListener,\
org.springframework.boot.context.config.ConfigFileApplicationListener,\
org.springframework.boot.context.config.DelegatingApplicationListener,\
org.springframework.boot.context.logging.ClasspathLoggingApplicationListener,\
org.springframework.boot.context.logging.LoggingApplicationListener,\
org.springframework.boot.liquibase.LiquibaseServiceLocatorApplicationListener
除了 ClearCachesApplicationListener 和 LiquibaseServiceLocatorApplicationListener 都实现了 spring 的接口 Ordered。
spring boot autoconfigure 中的监听
org.springframework.context.ApplicationListener=\
org.springframework.boot.autoconfigure.BackgroundPreinitializer
只有一个 BackgroundPreinitializer,通过注解 @Order 指定了顺序。
spring boot context 中的监听
data:image/s3,"s3://crabby-images/62690/62690356db43acd61beda8e121fe31d66806b419" alt=""
org.springframework.context.ApplicationListener=\
org.springframework.cloud.bootstrap.BootstrapApplicationListener,\
org.springframework.cloud.bootstrap.LoggingSystemShutdownListener,\
org.springframework.cloud.context.restart.RestartListener
都实现了 spring 的接口 Ordered。
将加载的监听进行排序
在 SpringApplication#getSpringFactoriesInstances() 中通过 AnnotationAwareOrderComparator.sort() 进行处理
data:image/s3,"s3://crabby-images/67873/678737ef40e7fd982b70ab45d8e7cda53a978c89" alt=""
其中排序逻辑在父类 OrderComparator 中进行实现。
data:image/s3,"s3://crabby-images/a6a89/a6a89118e1e55b349799c55511c35ad17dd3a62c" alt=""
data:image/s3,"s3://crabby-images/a0d72/a0d72de235d6a3e626df96ef7e8afe5ccdc7de1c" alt=""
如果当前监听实现了接口 Ordered,则按照对应的编号进行比较,否则直接返回最低优先级,即 Integer.MAX_VALUE。
data:image/s3,"s3://crabby-images/7819e/7819eb1c111d2c7789f21aa239ebc0f8c1cd85bf" alt=""
findOrder() 在子类 AnnotationAwareOrderComparator 中进行重写,用于处理使用注解 @Order 标记的类。
data:image/s3,"s3://crabby-images/24626/246263b9865b96fc0932eea82f52a1f0a92675a7" alt=""
spring boot 中的监听
对应的监听排序如下
CloudFoundryVcapEnvironmentPostProcessor
Integer.MIN_VALUE + 10 - 1
ConfigFileApplicationListener
Integer.MIN_VALUE + 10
AnsiOutputApplicationListener
Integer.MIN_VALUE + 10 + 1
LoggingApplicationListener
Integer.MIN_VALUE + 20
ClasspathLoggingApplicationListener
Integer.MIN_VALUE + 20 + 1
DelegatingApplicationListener
0
ParentContextCloserApplicationListener
Integer.MAX_VALUE - 10
FileEncodingApplicationListener
Integer.MAX_VALUE
ClearCachesApplicationListener
Integer.MAX_VALUE
LiquibaseServiceLocatorApplicationListener
Integer.MAX_VALUE
实际执行结果
data:image/s3,"s3://crabby-images/378b3/378b3ace4628ae8e5a78b2248ded31e65b2ed8bf" alt=""
spring boot context 中的监听
对应的监听排序如下
BootstrapApplicationListener
Integer.MIN_VALUE + 5
LoggingSystemShutdownListener
Integer.MIN_VALUE + 5 + 1
RestartListener
0
结合 spring boot 启动后的顺序如下
BootstrapApplicationListener
Integer.MIN_VALUE + 5
LoggingSystemShutdownListener
Integer.MIN_VALUE + 5 + 1
CloudFoundryVcapEnvironmentPostProcessor
Integer.MIN_VALUE + 10 - 1
ConfigFileApplicationListener
Integer.MIN_VALUE + 10
AnsiOutputApplicationListener
Integer.MIN_VALUE + 10 + 1
LoggingApplicationListener
Integer.MIN_VALUE + 20
ClasspathLoggingApplicationListener
Integer.MIN_VALUE + 20 + 1
DelegatingApplicationListener
0
RestartListener
0
ParentContextCloserApplicationListener
Integer.MAX_VALUE - 10
FileEncodingApplicationListener
Integer.MAX_VALUE
ClearCachesApplicationListener
Integer.MAX_VALUE
LiquibaseServiceLocatorApplicationListener
Integer.MAX_VALUE
实际执行结果
data:image/s3,"s3://crabby-images/78cd6/78cd602b1ba056aea8ada215203e49f5bfbb6e90" alt=""
可以看到,引入 spring cloud 组件后,默认执行 BootstrapApplicationListener 中的逻辑加载 bootstrap 相关的配置。
监听执行
监听加载到 SpringApplicationRunListeners 中
上面的监听先加载到了 SpringApplicationRunListeners,内部通过一个集合存储SpringApplicationRunListener 的实现类 EventPublishingRunListener 进行存储以进行后续处理。
data:image/s3,"s3://crabby-images/9e4b1/9e4b192f9c3b8c6c7539b41d6642322e5f665399" alt=""
通过 SpringApplication#getRunListeners() 进行加载
data:image/s3,"s3://crabby-images/d4923/d49237bfbf76976bbbe6479b3d9ae919105a4a65" alt=""
通过反射创建,构造器参数中传入当前类 SpringApplication。
data:image/s3,"s3://crabby-images/ad4ae/ad4aef4862d3c24bf96f43c58fc397006a9e7bbb" alt=""
将 SpringApplication 中加载的监听添加到接口 ApplicationEventMulticaster 的实现类。
SimpleApplicationEventMulticaster 的父类 AbstractApplicationEventMulticaster 的内部类 DefaultListenerRetriever 的变量 applicationListeners 中。
data:image/s3,"s3://crabby-images/84061/840614a65019114cdd4a3344c099bf4963fd94d9" alt=""
data:image/s3,"s3://crabby-images/85c64/85c6499f94a28c4f50b762902ba43220377432c2" alt=""
data:image/s3,"s3://crabby-images/3f8c2/3f8c2ea7b5fdff824678482def5976381b2b1076" alt=""
调用 SpringApplicationRunListeners 的 environmentPrepared() 进行监听调用
调用 SpringApplicationRunListeners#environmentPrepared() 间接调用 SpringApplicationRunListener 的 environmentPrepared(),对应的实现类为 EventPublishingRunListener。
执行 SpringApplication#prepareEnvironment()
data:image/s3,"s3://crabby-images/a1f12/a1f126bc594fecd7cad1ae7b17d6b5ee8496421c" alt=""
data:image/s3,"s3://crabby-images/a1486/a148654c7dd0eb874749b553feabe2c38d526b46" alt=""
调用 SimpleApplicationEventMulticaster 的 multicastEvent()
data:image/s3,"s3://crabby-images/795c8/795c8367297bea597a3f7c18ed338993b073cb9a" alt=""
调用 SpringApplicationRunListener#environmentPrepared()
data:image/s3,"s3://crabby-images/97fab/97fabe215772f93b874dccfa6f61882f45b2c922" alt=""
调用 SimpleApplicationEventMulticaster#multicastEvent(),参数为事件 ApplicationEnvironmentPreparedEvent。
data:image/s3,"s3://crabby-images/8b0d8/8b0d893a67010c4160f082016b810307736513eb" alt=""
循环遍历监听,看是否支持对应的事件
data:image/s3,"s3://crabby-images/03329/033299ed15746bf06dc05676b207a28ec889589b" alt=""
data:image/s3,"s3://crabby-images/3d531/3d53107a9133cfaf769e19d2d2867d947e2244fd" alt=""
由于排序靠前的监听中 ConfigFileApplicationListener 中引入了 ApplicationEnvironmentPreparedEvent,所以优先处理。
调用了 ConfigFileApplicationListener#onApplicationEvent() 后,处理步骤如下
data:image/s3,"s3://crabby-images/cbbfd/cbbfdc648782b20b8e3b5d2a1553bc0ce59f3282" alt=""
判断当前事件是否为 ApplicationEnvironmentPreparedEvent 的实例,符合条件,执行后续处理。
加载 classpath 中 META-INF/spring.factories 中指定的 EnvironmentPostProcessor 的实现类。
加载当前类为到 EnvironmentPostProcessor 中并进行排序,ConfigFileApplicationListener 实现了接口 EnvironmentPostProcessor。
执行 EnvironmentPostProcessor#postProcessEnvironment() 进行配置文件加载。
总结
综合上述表述,如果引入了 BootstrapApplicationListener 则优先加载,但是在源码中发现如下
data:image/s3,"s3://crabby-images/f1b22/f1b227b3edf3cd01833944243bc00641775fd072" alt=""
正好对应了我们平时写的 spring boot 启动类,可知,在启动时,如果引入了 spring cloud 组件,会先创建一个子容器来加载对应的配置,然后传递到父容器中进行参数传递,完成参数加载。