SpringApplication啟動流程

SpringApplication啟動流程

SpringApplication的啟動過程非常複雜,下面是在調用SpringApplication.run方法之後啟動的關鍵動作:

Spring Boot深入分析 - SpringApplication啟動流程

既然要了解SpringApplication的啟動流程,第一步當然是進入源碼裡看看:

public ConfigurableApplicationContext run(String... args) {

StopWatch stopWatch = new StopWatch();

stopWatch.start();

ConfigurableApplicationContext context = null;

FailureAnalyzers analyzers = null;

configureHeadlessProperty();

// 初始化監聽器

SpringApplicationRunListeners listeners = getRunListeners(args);

// 發佈ApplicationStartEvent

listeners.starting();

try {

// 裝配參數和環境

ApplicationArguments applicationArguments = new DefaultApplicationArguments(

args);

ConfigurableEnvironment environment = prepareEnvironment(listeners,

applicationArguments);

// 打印 Banner

Banner printedBanner = printBanner(environment);

// 創建 ApplicationContext()

context = createApplicationContext();

analyzers = new FailureAnalyzers(context);

// 裝配 Context

prepareContext(context, environment, listeners, applicationArguments,

printedBanner);

// refresh Context

refreshContext(context);

// after Refresh

afterRefresh(context, applicationArguments);

// 發佈ApplicationReadyEvent

listeners.finished(context, null);

stopWatch.stop();

if (this.logStartupInfo) {

new StartupInfoLogger(this.mainApplicationClass)

.logStarted(getApplicationLog(), stopWatch);

}

return context;

}

catch (Throwable ex) {

handleRunFailure(context, listeners, analyzers, ex);

throw new IllegalStateException(ex);

}

}

第一步,初始化監聽器

這裡會初始化Spring Boot自帶的監聽器,以及添加到SpringApplication的自定義監聽器。

初始化監聽器的調用關係很深,為了節省篇幅,就不貼源碼了。

第二步,發佈ApplicationStartedEvent事件

到這一步,Spring Boot會發佈一個ApplicationStartedEvent事件。如果你想在這個時候執行一些代碼可以通過實現ApplicationListener接口實現;

下面是ApplicationListener接口的定義,注意這裡有個

public interface ApplicationListener extends EventListener

例如,你想監聽ApplicationStartedEvent事件,你可以這樣定義:

public class ApplicationStartedListener implements ApplicationListener<applicationstartedevent>

然後通過SpringApplication.addListener(..)添加進去即可。

第三步,裝配參數和環境

在這一步,首先會初始化參數,然後裝配環境,確定是web環境還是非web環境。

第四步,發佈ApplicationEnvironmentPreparedEvent事件

準確的說,這個應該屬於第三步,在裝配完環境後,就觸發ApplicationEnvironmentPreparedEvent事件。如果想在這個時候執行一些代碼,可以訂閱這個事件的監聽器,方法同第二步。

第五步,打印Banner

看過Spring Boot實例教程 - 自定義Banner的同學會很熟悉,啟動的Banner就是在這一步打印出來的。

第六步,創建ApplicationContext

這裡會根據是否是web環境,來決定創建什麼類型的ApplicationContext,ApplicationContext不要多說了吧,不知道ApplicationContext是啥的同學,出門左轉補下Spring基礎知識吧。

第七步,裝配Context

這裡會設置Context的環境變量、註冊Initializers、beanNameGenerator等。

第八步,發佈ApplicationPreparedEvent事件

這裡放在第七步會更準確,因為這個是在裝配Context的時候發佈的。

值得注意的是:這裡是假的,假的,假的,源碼中是空的,並沒有真正發佈ApplicationPreparedEvent事件。不知道作者這麼想的???

第九步,註冊、加載等

註冊springApplicationArguments、springBootBanner,加載資源等。

第十步,發佈ApplicationPreparedEvent事件

注意,到這裡才是真正發佈了ApplicationPreparedEvent事件。這裡和第八步好讓人誤解。

第十一步,refreshContext

裝配context beanfactory等非常重要的核心組件。

第十二步,afterRefreshContext

這裡會調用自定義的Runners,不知道Runners是什麼的同學,請參考Spring Boot官方文檔 - SpringApplication

第十三步,發佈ApplicationReadyEvent事件

最後一步,發佈ApplicationReadyEvent事件,啟動完畢,表示服務已經可以開始正常提供服務了。通常我們這裡會監聽這個事件來打印一些監控性質的日誌,表示應用正常啟動了。添加方法同第二步。

注意:如果啟動失敗,這一步會發布ApplicationFailedEvent事件。

到這裡,Spring Boot啟動的一些關鍵動作就介紹完了。


分享到:


相關文章: