SpringBoot中文注釋項目Github地址:
https://github.com/yuanmabiji/spring-boot-2.1.0.RELEASE
本篇接 SpringBoot事件監聽機制源碼分析(上) SpringBoot源碼(九)
1 溫故而知新
溫故而知新,我們來簡單回顧一下上篇的內容,上一篇我們分析了SpringBoot啟動時廣播生命周期事件的原理,現將關鍵步驟再濃縮總結下:
- 為廣播SpringBoot內置生命周期事件做前期準備:1)首先加載
ApplicationListener
監聽器實現類;2)其次加載SPI擴展類EventPublishingRunListener
。 - SpringBoot啟動時利用
EventPublishingRunListener
廣播生命周期事件,然后ApplicationListener
監聽器實現類監聽相應的生命周期事件執行一些初始化邏輯的工作。
2 引言
上篇文章的側重點是分析了SpringBoot啟動時廣播生命周期事件的原理,此篇文章我們再來詳細分析SpringBoot內置的7種生命周期事件的源碼。
3 SpringBoot生命周期事件源碼分析
分析SpringBoot的生命周期事件,我們先來看一張類結構圖:
由上圖可以看到事件類之間的關系:
- 最頂級的父類是JDK的事件基類
EventObject
; - 然后Spring的事件基類
ApplicationEvent
繼承了JDK的事件基類EventObject
; - 其次SpringBoot的生命周期事件基類
SpringApplicationEvent
繼承了Spring的事件基類ApplicationEvent
; - 最后SpringBoot具體的7個生命周期事件類再繼承了SpringBoot的生命周期事件基類
SpringApplicationEvent
。
3.1 JDK的事件基類EventObject
EventObject
類是JDK的事件基類,可以說是所有Java事件類的基本,即所有的Java事件類都直接或間接繼承于該類,源碼如下:
// EventObject.java
public class EventObject implements java.io.Serializable {
private static final long serialVersionUID = 5516075349620653480L;
/**
* The object on which the Event initially occurred.
*/
protected transient Object source;
/**
* Constructs a prototypical Event.
*
* @param source The object on which the Event initially occurred.
* @exception IllegalArgumentException if source is null.
*/
public EventObject(Object source) {
if (source == null)
throw new IllegalArgumentException("null source");
this.source = source;
}
/**
* The object on which the Event initially occurred.
*
* @return The object on which the Event initially occurred.
*/
public Object getSource() {
return source;
}
/**
* Returns a String representation of this EventObject.
*
* @return A a String representation of this EventObject.
*/
public String toString() {
return getClass().getName() + "[source=" + source + "]";
}
}
可以看到EventObject
類只有一個屬性source
,這個屬性是用來記錄最初事件是發生在哪個類,舉個栗子,比如在SpringBoot啟動過程中會發射ApplicationStartingEvent
事件,而這個事件最初是在SpringApplication
類中發射的,因此source
就是SpringApplication
對象。
3.2 Spring的事件基類ApplicationEvent
ApplicationEvent
繼承了DK的事件基類EventObject
類,是Spring的事件基類,被所有Spring的具體事件類繼承,源碼如下:
// ApplicationEvent.java
/**
* Class to be extended by all application events. Abstract as it
* doesn't make sense for generic events to be published directly.
*
* @author Rod Johnson
* @author Juergen Hoeller
*/
public abstract class ApplicationEvent extends EventObject {
/** use serialVersionUID from Spring 1.2 for interoperability. */
private static final long serialVersionUID = 7099057708183571937L;
/** System time when the event happened. */
private final long timestamp;
/**
* Create a new ApplicationEvent.
* @param source the object on which the event initially occurred (never {@code null})
*/
public ApplicationEvent(Object source) {
super(source);
this.timestamp = System.currentTimeMillis();
}
/**
* Return the system time in milliseconds when the event happened.
*/
public final long getTimestamp() {
return this.timestamp;
}
}
可以看到ApplicationEvent
有且僅有一個屬性timestamp
,該屬性是用來記錄事件發生的時間。
3.3 SpringBoot的事件基類SpringApplicationEvent
SpringApplicationEvent
類繼承了Spring的事件基類ApplicationEvent
,是所有SpringBoot內置生命周期事件的父類,源碼如下:
/**
* Base class for {@link ApplicationEvent} related to a {@link SpringApplication}.
*
* @author Phillip Webb
*/
@SuppressWarnings("serial")
public abstract class SpringApplicationEvent extends ApplicationEvent {
private final String[] args;
public SpringApplicationEvent(SpringApplication application, String[] args) {
super(application);
this.args = args;
}
public SpringApplication getSpringApplication() {
return (SpringApplication) getSource();
}
public final String[] getArgs() {
return this.args;
}
}
可以看到SpringApplicationEvent
有且僅有一個屬性args
,該屬性就是SpringBoot啟動時的命令行參數即標注@SpringBootApplication
啟動類中main
函數的參數。
3.4 SpringBoot具體的生命周期事件類
接下來我們再來看一下SpringBoot
內置生命周期事件即SpringApplicationEvent
的具體子類們。
3.4.1 ApplicationStartingEvent
// ApplicationStartingEvent.java
public class ApplicationStartingEvent extends SpringApplicationEvent {
public ApplicationStartingEvent(SpringApplication application, String[] args) {
super(application, args);
}
}
SpringBoot開始啟動時便會發布ApplicationStartingEvent
事件,其發布時機在環境變量Environment或容器ApplicationContext創建前但在注冊ApplicationListener
具體監聽器之后,標志標志SpringApplication
開始啟動。
3.4.2 ApplicationEnvironmentPreparedEvent
// ApplicationEnvironmentPreparedEvent.java
public class ApplicationEnvironmentPreparedEvent extends SpringApplicationEvent {
private final ConfigurableEnvironment environment;
/**
* Create a new {@link ApplicationEnvironmentPreparedEvent} instance.
* @param application the current application
* @param args the arguments the application is running with
* @param environment the environment that was just created
*/
public ApplicationEnvironmentPreparedEvent(SpringApplication application,
String[] args, ConfigurableEnvironment environment) {
super(application, args);
this.environment = environment;
}
/**
* Return the environment.
* @return the environment
*/
public ConfigurableEnvironment getEnvironment() {
return this.environment;
}
}
可以看到ApplicationEnvironmentPreparedEvent
事件多了一個environment
屬性,我們不妨想一下,多了environment
屬性的作用是啥?
答案就是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
幾乎已經完全準備好了。
思考: 監聽同一事件的監聽器們執行監聽邏輯時是有順序的,我們可以想一下這個排序邏輯是什么時候排序的?還有為什么要這樣排序呢?
3.4.3 ApplicationContextInitializedEvent
// ApplicationContextInitializedEvent.java
public class ApplicationContextInitializedEvent extends SpringApplicationEvent {
private final ConfigurableApplicationContext context;
/**
* Create a new {@link ApplicationContextInitializedEvent} instance.
* @param application the current application
* @param args the arguments the application is running with
* @param context the context that has been initialized
*/
public ApplicationContextInitializedEvent(SpringApplication application,
String[] args, ConfigurableApplicationContext context) {
super(application, args);
this.context = context;
}
/**
* Return the application context.
* @return the context
*/
public ConfigurableApplicationContext getApplicationContext() {
return this.context;
}
}
可以看到ApplicationContextInitializedEvent
事件多了個ConfigurableApplicationContext
類型的context
屬性,context
屬性的作用同樣是為了相應監聽器可以拿到這個context
屬性執行一些邏輯,具體作用將在3.4.4
詳述。
ApplicationContextInitializedEvent
事件在ApplicationContext
容器創建后,且為ApplicationContext
容器設置了environment
變量和執行了ApplicationContextInitializers
的初始化方法后但在bean定義加載前觸發,標志ApplicationContext已經初始化完畢。
擴展: 可以看到
ApplicationContextInitializedEvent
是在為context
容器配置environment
變量后觸發,此時ApplicationContextInitializedEvent
等事件只要有context
容器的話,那么其他需要environment
環境變量的監聽器只需要從context
中取出environment
變量即可,從而ApplicationContextInitializedEvent
等事件沒必要再配置environment
屬性。
3.4.4 ApplicationPreparedEvent
// ApplicationPreparedEvent.java
public class ApplicationPreparedEvent extends SpringApplicationEvent {
private final ConfigurableApplicationContext context;
/**
* Create a new {@link ApplicationPreparedEvent} instance.
* @param application the current application
* @param args the arguments the application is running with
* @param context the ApplicationContext about to be refreshed
*/
public ApplicationPreparedEvent(SpringApplication application, String[] args,
ConfigurableApplicationContext context) {
super(application, args);
this.context = context;
}
/**
* Return the application context.
* @return the context
*/
public ConfigurableApplicationContext getApplicationContext() {
return this.context;
}
}
同樣可以看到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
已經準備好可以用了。
3.4.5 ApplicationStartedEvent
// ApplicationStartedEvent.java
public class ApplicationStartedEvent extends SpringApplicationEvent {
private final ConfigurableApplicationContext context;
/**
* Create a new {@link ApplicationStartedEvent} instance.
* @param application the current application
* @param args the arguments the application is running with
* @param context the context that was being created
*/
public ApplicationStartedEvent(SpringApplication application, String[] args,
ConfigurableApplicationContext context) {
super(application, args);
this.context = context;
}
/**
* Return the application context.
* @return the context
*/
public ConfigurableApplicationContext getApplicationContext() {
return this.context;
}
}
ApplicationStartedEvent
事件將在容器刷新后但ApplicationRunner
和CommandLineRunner
的run
方法執行前觸發,標志Spring
容器已經刷新,此時容器已經準備完畢了。
擴展: 這里提到了
ApplicationRunner
和CommandLineRunner
接口有啥作用呢?我們一般會在Spring
容器刷新完畢后,此時可能有一些系統參數等靜態數據需要加載,此時我們就可以實現了ApplicationRunner
或CommandLineRunner
接口來實現靜態數據的加載。
3.4.6 ApplicationReadyEvent
// ApplicationReadyEvent.java
public class ApplicationReadyEvent extends SpringApplicationEvent {
private final ConfigurableApplicationContext context;
/**
* Create a new {@link ApplicationReadyEvent} instance.
* @param application the current application
* @param args the arguments the application is running with
* @param context the context that was being created
*/
public ApplicationReadyEvent(SpringApplication application, String[] args,
ConfigurableApplicationContext context) {
super(application, args);
this.context = context;
}
/**
* Return the application context.
* @return the context
*/
public ConfigurableApplicationContext getApplicationContext() {
return this.context;
}
}
ApplicationReadyEvent
事件在調用完ApplicationRunner
和CommandLineRunner
的run
方法后觸發,此時標志SpringApplication
已經正在運行。
3.4.7 ApplicationFailedEvent
// ApplicationFailedEvent.java
public class ApplicationFailedEvent extends SpringApplicationEvent {
private final ConfigurableApplicationContext context;
private final Throwable exception;
/**
* Create a new {@link ApplicationFailedEvent} instance.
* @param application the current application
* @param args the arguments the application was running with
* @param context the context that was being created (maybe null)
* @param exception the exception that caused the error
*/
public ApplicationFailedEvent(SpringApplication application, String[] args,
ConfigurableApplicationContext context, Throwable exception) {
super(application, args);
this.context = context;
this.exception = exception;
}
/**
* Return the application context.
* @return the context
*/
public ConfigurableApplicationContext getApplicationContext() {
return this.context;
}
/**
* Return the exception that caused the failure.
* @return the exception
*/
public Throwable getException() {
return this.exception;
}
}
可以看到ApplicationFailedEvent
事件除了多了一個context
屬性外,還多了一個Throwable
類型的exception
屬性用來記錄SpringBoot啟動失敗時的異常。
ApplicationFailedEvent
事件在SpringBoot啟動失敗時觸發,標志SpringBoot啟動失敗。
4 小結
此篇文章相對簡單,對SpringBoot內置的7種生命周期事件進行了詳細分析。我們還是引用上篇文章的一張圖來回顧一下這些生命周期事件及其用途:
5 寫在最后
由于有一些小伙伴們建議之前有些源碼分析文章太長,導致耐心不夠,看不下去,因此,之后的源碼分析文章如果太長的話,筆者將會考慮拆分為幾篇文章,這樣就比較短小了,比較容易看完,嘿嘿。
【源碼筆記】Github地址:
https://github.com/yuanmabiji/Java-SourceCode-Blogs
點贊搞起來,嘿嘿嘿!
公眾號【源碼筆記】專注于Java后端系列框架的源碼分析。