Spring Boot 啟動流程解析
Spring Boot 是一個簡化的 Spring 應(yīng)用開發(fā)框架,它以 “約定優(yōu)于配置” 的理念,為開發(fā)者提供了開箱即用的功能。在 Spring Boot 的運(yùn)行過程中,SpringApplication.run()
方法是整個啟動流程的核心,本文將詳細(xì)剖析其內(nèi)部實(shí)現(xiàn),幫助你深入理解 Spring Boot 的啟動機(jī)制。
1. Spring Boot 啟動的主要步驟
Spring Boot 的啟動由 SpringApplication.run()
方法驅(qū)動,其核心代碼如下:
SpringApplication.run(Application.class, args);
這個方法的運(yùn)行過程可以分為以下主要步驟:
初始化 SpringApplication
確定應(yīng)用類型(Servlet/Reactive/非 Web),加載環(huán)境和監(jiān)聽器。
準(zhǔn)備運(yùn)行環(huán)境
加載 application.properties
或 application.yml
配置,解析命令行參數(shù),準(zhǔn)備 ConfigurableEnvironment
。
創(chuàng)建應(yīng)用上下文
根據(jù)應(yīng)用類型創(chuàng)建對應(yīng)的 ApplicationContext
,例如:
AnnotationConfigServletWebServerApplicationContext
(Servlet 應(yīng)用)ReactiveWebServerApplicationContext
(Reactive 應(yīng)用)
刷新上下文
完成 Bean 的加載、依賴注入、AOP 代理等操作,啟動內(nèi)嵌的 Web 服務(wù)器(如果需要)。
通知監(jiān)聽器和執(zhí)行擴(kuò)展點(diǎn)
觸發(fā)生命周期事件,執(zhí)行 CommandLineRunner
和 ApplicationRunner
。
啟動完成
應(yīng)用準(zhǔn)備好后,觸發(fā) ApplicationReadyEvent
,正式進(jìn)入運(yùn)行狀態(tài)。
2. 核心組件解析
2.1 環(huán)境準(zhǔn)備
Spring Boot 通過 ConfigurableEnvironment
統(tǒng)一管理環(huán)境變量、配置文件、系統(tǒng)屬性等信息。
加載順序:
- 系統(tǒng)屬性(如
-D
參數(shù))。 - 環(huán)境變量(
System.getenv()
)。 - 配置文件(
application.properties
或application.yml
)。 - 命令行參數(shù)(
args
)。
擴(kuò)展點(diǎn):
可以通過實(shí)現(xiàn) EnvironmentPostProcessor
來自定義環(huán)境變量加載邏輯。
2.2 上下文創(chuàng)建
Spring Boot 根據(jù)應(yīng)用類型選擇合適的 ApplicationContext
:
應(yīng)用類型 | 上下文類型 | 說明 |
---|---|---|
Servlet 應(yīng)用 | AnnotationConfigServletWebServerApplicationContext | 默認(rèn)的 Web 應(yīng)用上下文。 |
Reactive 應(yīng)用 | ReactiveWebServerApplicationContext | 響應(yīng)式編程應(yīng)用上下文。 |
非 Web 應(yīng)用 | GenericApplicationContext | 普通 Java 應(yīng)用上下文。 |
上下文創(chuàng)建時,還會完成以下工作:
- 注冊基礎(chǔ)組件,如
Environment
和ResourceLoader
。 - 加載 Bean 定義,準(zhǔn)備好應(yīng)用的依賴圖。
2.3 Bean 加載與初始化
掃描與加載:
通過 @ComponentScan
掃描包路徑,加載所有標(biāo)注了 @Component
、@Service
、@Repository
、@Controller
等注解的類。
生命周期回調(diào):
初始化過程中,Spring 會調(diào)用以下接口或注解:
InitializingBean
的afterPropertiesSet()
。@PostConstruct
方法。
AOP 代理:
根據(jù) @EnableAspectJAutoProxy
自動為符合條件的 Bean 創(chuàng)建代理。
2.4 自動配置
Spring Boot 的 “魔法” 很大程度上來自于自動配置機(jī)制,通過 @EnableAutoConfiguration
實(shí)現(xiàn)。
實(shí)現(xiàn)原理:
Spring Boot 使用 spring.factories
文件加載所有自動配置類。
核心類:@ConditionalOnClass
、@ConditionalOnProperty
等條件注解,控制是否加載自動配置。
擴(kuò)展點(diǎn):
可以通過實(shí)現(xiàn) AutoConfigurationImportFilter
自定義自動配置類的加載行為。
3. 生命周期事件
Spring Boot 在啟動過程中會觸發(fā)以下事件,開發(fā)者可以通過監(jiān)聽器進(jìn)行自定義邏輯處理:
例如,監(jiān)聽 ApplicationReadyEvent
:
事件名稱 | 觸發(fā)時機(jī) |
---|---|
ApplicationStartingEvent | 應(yīng)用啟動開始。 |
ApplicationEnvironmentPreparedEvent | 環(huán)境變量準(zhǔn)備完成。 |
ApplicationPreparedEvent | 上下文準(zhǔn)備完成但未刷新。 |
ApplicationStartedEvent | 上下文刷新完成。 |
ApplicationReadyEvent | 應(yīng)用完全啟動,準(zhǔn)備好接收請求。 |
ApplicationFailedEvent | 啟動失敗。 |
例如,監(jiān)聽 ApplicationReadyEvent
:
@Component public class MyApplicationReadyListener implements ApplicationListener<ApplicationReadyEvent> { @Override public void onApplicationEvent(ApplicationReadyEvent event) { System.out.println("Application is ready!"); } }
4. 啟動異常處理機(jī)制
Spring Boot 使用 handleRunFailure
方法集中處理啟動過程中的異常:
處理方式:
- 打印錯誤日志。
- 調(diào)用注冊的
SpringApplicationRunListener
的failed()
方法。 - 關(guān)閉應(yīng)用上下文,釋放資源。
常見異常:
BeanDefinitionStoreException
:Bean 定義加載錯誤。UnsatisfiedDependencyException
:依賴注入失敗。IllegalStateException
:上下文狀態(tài)非法。
5. 總結(jié)
Spring Boot 的啟動流程高度模塊化,既有固定的主流程,又為開發(fā)者提供了豐富的擴(kuò)展點(diǎn)。主要流程可總結(jié)為:
- 準(zhǔn)備階段:初始化
SpringApplication
,加載環(huán)境和監(jiān)聽器。 - 上下文加載:創(chuàng)建
ApplicationContext
,加載配置和 Bean 定義。 - 上下文刷新:完成 Bean 的依賴注入和初始化,啟動內(nèi)嵌服務(wù)器。
- 擴(kuò)展與回調(diào):通知監(jiān)聽器、調(diào)用
Runner
接口、觸發(fā)生命周期事件。
通過理解 Spring Boot 啟動流程,你可以更高效地調(diào)試問題、優(yōu)化啟動性能,甚至自定義啟動行為,從而更好地掌控你的應(yīng)用。
到此這篇關(guān)于Spring Boot 啟動流程詳解的文章就介紹到這了,更多相關(guān)Spring Boot 啟動流程內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
SpringBoot application.yml和bootstrap.yml的區(qū)別
本文主要介紹了SpringBoot application.yml和bootstrap.yml的區(qū)別,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2023-04-04Java中IdentityHashMap與HashMap區(qū)別詳解
這篇文章主要介紹了Java中IdentityHashMap與HashMap區(qū)別詳解,很多人不曉得IdentityHashMap的存在,其中不乏工作很多年的Java開發(fā)者,他們看到就說是第三方j(luò)ar包,實(shí)際上它是Jdk源碼自帶的集合類,需要的朋友可以參考下2023-11-11MyBatis實(shí)現(xiàn)兩種查詢樹形數(shù)據(jù)的方法詳解(嵌套結(jié)果集和遞歸查詢)
樹形結(jié)構(gòu)數(shù)據(jù)在開發(fā)中十分常見,比如:菜單數(shù)、組織樹, 利用 MyBatis 提供嵌套查詢功能可以很方便地實(shí)現(xiàn)這個功能需求。本文主要介紹了兩種方法,感興趣的可以了解一下2021-09-09mybatisplus解除分頁限制的實(shí)現(xiàn)
這篇文章主要介紹了mybatisplus解除分頁限制的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-12-12springboot打包部署到linux服務(wù)器的方法
這篇文章主要介紹了springboot打包部署到linux服務(wù)器的方法,通過實(shí)例代碼相結(jié)合的形式給大家介紹的非常詳細(xì),具有一定的參考借鑒價值,需要的朋友可以參考下2018-06-06