0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

SpringBoot的嵌入式Web容器是什么時候加載的?

jf_ro2CN3Fa ? 來源:CSDN ? 2023-07-11 10:10 ? 次閱讀

0.前言

背景:最近有位開發(fā)同學(xué)說面試被問到Spring Boot 的啟動流程,以及被問到Spring Boot 的嵌入式Web容器是什么時候加載的。如何加載的。是怎么無縫切換的。

這些問題,其實回答起來也是比較復(fù)雜的。我們今天就從 SpringApplication.run(EasyPaasAdminApplication.class, args);入口,逐漸向下看下執(zhí)行流程。來試著回答一下前面這兩個問題。

后面關(guān)于SpringBoot 的web容器可以無縫隨意切換為jetty,undertow..這個問題的回答涉及到Spring Boot是如何設(shè)計WebServer的。我們后續(xù)專門講解一下。

1. 執(zhí)行邏輯梳理

一般我們SpringBoot 應(yīng)用的啟動入口都是如下這種固定的寫法,

de939068-1f8a-11ee-962d-dac502259ad0.png

也可以是這樣

publicstaticvoidmain(String[]args){
SpringApplicationapplication=newSpringApplication(MyApplication.class);
//...customizeapplicationsettingshere
application.run(args)
}

但總之,都是使用SpringApplication 調(diào)用靜態(tài)方法

此方法的注釋

Static helper that can be used to run a SpringApplication from the specified source using default settings.

publicstaticConfigurableApplicationContextrun(ClassprimarySource,String...args){
returnrun(newClass[]{primarySource},args);
}

跟過來就到這,可以看到注釋運行Spring應(yīng)用程序,創(chuàng)建并刷新一個新的ApplicationContext。

dece32cc-1f8a-11ee-962d-dac502259ad0.pngdeedf9ea-1f8a-11ee-962d-dac502259ad0.png

跟代碼到這兒其實我們對于SpringBoot 的基本啟動流程已經(jīng)知道了。但是要解答什么時候啟動的Tomcat 還需要繼續(xù)分析。

df2549ea-1f8a-11ee-962d-dac502259ad0.png

到這兒我們就可以繼續(xù)下去,發(fā)現(xiàn)Spring Boot 啟動WebServer。此處的WebServer我就不展開了,可以點擊去就三個方法start ,stop,getPort。可以看出來Spring 在設(shè)計接口的時候還是很嚴(yán)謹(jǐn)和精簡。

我們的核心脈絡(luò)是梳理SpringBoot 啟動過程,并且回答Tomcat 是如何被啟動的。

df49063c-1f8a-11ee-962d-dac502259ad0.png

我們可以看到WebServer 的實現(xiàn)目前內(nèi)置的有5種。其實Spring Boot 還有一個特性叫做 自動裝配。

這就是為什么5個實現(xiàn),我們最后啟動的是Tomcat。此處也不做展開。后面我專門搞一個解析SpringBoot 自動裝配的文章。

df7fd20c-1f8a-11ee-962d-dac502259ad0.png

我們看一下內(nèi)部start 的TomcatWebServer的內(nèi)部實現(xiàn)。了解過Tomcat 源碼的同學(xué)看到這兒就基本明白了。

dfa25a34-1f8a-11ee-962d-dac502259ad0.png

好源碼跟進過程我們到此結(jié)束,我們整理和總結(jié)一下。

通過掃一遍源碼我們大概可以總結(jié)出來如下三個階段

準(zhǔn)備階段、應(yīng)用上下文創(chuàng)建階段、刷新上下文階段。

準(zhǔn)備階段 :Spring Boot 會加載應(yīng)用程序的初始設(shè)置,并創(chuàng)建 Spring Boot 上下文。這個階段的核心源碼是 SpringApplication 類的 run() 方法,它會調(diào)用 Spring Boot 的各個初始化器進行初始化和準(zhǔn)備工作。

應(yīng)用上下文創(chuàng)建階段 : Spring Boot 會創(chuàng)建應(yīng)用程序的上下文,包括各種配置信息、Bean 的加載和初始化等。這個階段的核心源碼是 Spring Boot 自動配置機制,通過掃描 classpath 中的配置文件,自動加載和配置各種組件和 Bean。

刷新上下文階段 :Spring Boot 會執(zhí)行各種啟動任務(wù),包括創(chuàng)建 Web 服務(wù)器、加載應(yīng)用程序的配置、初始化各種組件等。這個階段的核心源碼是 Spring Boot 的刷新機制,它會調(diào)用各種初始化器和監(jiān)聽器,執(zhí)行各種啟動任務(wù)。其中啟動Tomcat 就是在這個環(huán)節(jié)進行。

2. 核心源碼解析

既然上面我們已經(jīng)基本上總結(jié)除了,Spring Boot的啟動脈絡(luò)。也梳理出了一些核心源碼。那么我們對啟動過程的核心源碼解析一下。

2.1. 準(zhǔn)備階段

在準(zhǔn)備階段中,Spring Boot 會加載應(yīng)用程序的初始設(shè)置,并創(chuàng)建 Spring Boot 上下文。這個階段的核心源碼是 SpringApplication 類的 run() 方法,它會調(diào)用 Spring Boot 的各個初始化器進行初始化和準(zhǔn)備工作。

publicConfigurableApplicationContextrun(String...args){
//啟動計時器
StopWatchstopWatch=newStopWatch();
stopWatch.start();

//定義應(yīng)用程序上下文和異常報告器列表
ConfigurableApplicationContextcontext=null;
CollectionexceptionReporters=newArrayList<>();

//配置Headless屬性
configureHeadlessProperty();

//獲取SpringBoot啟動監(jiān)聽器
SpringApplicationRunListenerslisteners=getRunListeners(args);
//執(zhí)行啟動監(jiān)聽器的starting方法
listeners.starting();

try{
//解析命令行參數(shù)
ApplicationArgumentsapplicationArguments=newDefaultApplicationArguments(args);
//準(zhǔn)備應(yīng)用程序環(huán)境
ConfigurableEnvironmentenvironment=prepareEnvironment(listeners,applicationArguments);
//配置忽略BeanInfo
configureIgnoreBeanInfo(environment);
//打印Banner
BannerprintedBanner=printBanner(environment);
//創(chuàng)建應(yīng)用程序上下文
context=createApplicationContext();
//獲取異常報告器,關(guān)于異常報告,我下次專門講一下SpringBoot 的異常收集器。
exceptionReporters=getSpringFactoriesInstances(SpringBootExceptionReporter.class,newClass[]{ConfigurableApplicationContext.class},context);
//準(zhǔn)備應(yīng)用程序上下文
prepareContext(context,environment,listeners,applicationArguments,printedBanner);
//刷新應(yīng)用程序上下文
refreshContext(context);
//刷新后操作
afterRefresh(context,applicationArguments);
//停止計時器
stopWatch.stop();
//記錄啟動日志
if(this.logStartupInfo){
newStartupInfoLogger(this.mainApplicationClass).logStarted(getApplicationLog(),stopWatch);
}
//執(zhí)行啟動監(jiān)聽器的started方法
listeners.started(context);
//執(zhí)行Runner
callRunners(context,applicationArguments);
}catch(Throwableex){
//處理啟動失敗
handleRunFailure(context,ex,exceptionReporters,listeners);
thrownewIllegalStateException(ex);
}

try{
//執(zhí)行啟動監(jiān)聽器的running方法
listeners.running(context);
}catch(Throwableex){
//處理啟動失敗
handleRunFailure(context,ex,exceptionReporters,null);
thrownewIllegalStateException(ex);
}

//返回應(yīng)用程序上下文
returncontext;
}

在 run() 方法中,Spring Boot 首先會創(chuàng)建一個 StopWatch 對象,用于記錄整個啟動過程的耗時。然后,Spring Boot 會調(diào)用 getRunListeners(args) 方法獲取 Spring Boot 的各個啟動監(jiān)聽器,并調(diào)用starting() 方法通知這些監(jiān)聽器啟動過程已經(jīng)開始。接著調(diào)用 prepareEnvironment(listeners, applicationArguments) 方法創(chuàng)建應(yīng)用程序的環(huán)境變量。

這個方法會根據(jù)用戶的配置和默認(rèn)設(shè)置創(chuàng)建一個 ConfigurableEnvironment對象,并將其傳給后面的 createApplicationContext() 方法。printBanner(environment) 方法打印啟動界面的 Banner,調(diào)用 refreshContext(context)方法刷新上下文。這個方法會啟動上下文,執(zhí)行各種啟動任務(wù),包括創(chuàng)建 Web 服務(wù)器、加載應(yīng)用程序的配置、初始化各種組件等。具體的啟動任務(wù)會在刷新上下文階段中進行。

2.2. 應(yīng)用上下文創(chuàng)建階段

在應(yīng)用上下文創(chuàng)建階段中,Spring Boot 會創(chuàng)建應(yīng)用程序的上下文,包括各種配置信息、Bean 的加載和初始化等。這個階段的核心源碼是 Spring Boot 自動配置機制,通過掃描 classpath 中的配置文件,自動加載和配置各種組件和 Bean。

protectedConfigurableApplicationContextcreateApplicationContext(){
ClasscontextClass=this.applicationContextClass;
if(contextClass==null){
try{
switch(this.webApplicationType){
caseSERVLET:
contextClass=Class.forName(DEFAULT_SERVLET_WEB_CONTEXT_CLASS);
break;
caseREACTIVE:
contextClass=Class.forName(DEFAULT_REACTIVE_WEB_CONTEXT_CLASS);
break;
default:
contextClass=Class.forName(DEFAULT_CONTEXT_CLASS);
}
}
catch(ClassNotFoundExceptionex){
thrownewIllegalStateException(
"UnabletocreateadefaultApplicationContext,"+
"pleasespecifyanApplicationContextClass",ex);
}
}
return(ConfigurableApplicationContext)BeanUtils.instantiateClass(contextClass);
}

在 createApplicationContext() 方法中,Spring Boot 首先會判斷應(yīng)用程序的類型,如果是 Web 應(yīng)用程序,則會創(chuàng)建一個 WebApplicationContext;否則,會創(chuàng)建一個普通的 ApplicationContext。調(diào)用 BeanUtils.instantiateClass(contextClass) 方法創(chuàng)建應(yīng)用程序的上下文。這個方法會根據(jù)上面的邏輯創(chuàng)建一個相應(yīng)的 ApplicationContext。調(diào)用 load() 方法加載應(yīng)用程序的配置。

這個方法會掃描 classpath 中的各種配置文件,例如 application.properties、application.yml、META-INF/spring.factories 等,自動配置各種組件和 Bean。調(diào)用 postProcessApplicationContext() 方法對應(yīng)用程序的上下文進行后處理。這個方法會調(diào)用各種初始化器和監(jiān)聽器,執(zhí)行各種初始化任務(wù)。

2.3. 刷新上下文階段

在刷新上下文階段中,Spring Boot 會執(zhí)行各種啟動任務(wù),包括創(chuàng)建 Web 服務(wù)器(剛才我們跟源碼的時候也看到了,如上我的截圖)、加載應(yīng)用程序的配置、初始化各種組件等。這個階段的核心源碼是 Spring Boot 的刷新機制,它會調(diào)用各種初始化器和監(jiān)聽器,執(zhí)行各種啟動任務(wù)。

protectedvoidrefreshContext(ConfigurableApplicationContextapplicationContext){
refresh(applicationContext);
if(this.registerShutdownHook){
try{
applicationContext.registerShutdownHook();
}
catch(AccessControlExceptionex){
//Notallowedinsomeenvironments.
}
}
}

在 refreshContext() 方法中調(diào)用 refresh(applicationContext) 方法刷新上下文。這個方法是 ApplicationContext 接口的核心方法,會啟動上下文,執(zhí)行各種啟動任務(wù)。調(diào)用 registerShutdownHook() 方法注冊應(yīng)用程序的關(guān)閉鉤子。這個方法會在應(yīng)用程序關(guān)閉時自動執(zhí)行,清理資源、關(guān)閉線程等,所以我們利用此特性在服務(wù)關(guān)閉的時候清理一些資源。并向外部發(fā)送告警通知。

在 refresh(applicationContext) 方法中,Spring Boot 會執(zhí)行上下文的各種啟動任務(wù),包括創(chuàng)建 Web 服務(wù)器、加載應(yīng)用程序的配置、初始化各種組件等。具體的啟動任務(wù)會調(diào)用各種初始化器和監(jiān)聽器,例如:

for(ApplicationContextInitializerinitializer:getInitializers()){
initializer.initialize(applicationContext);
}

另外,Spring Boot 還會調(diào)用各種監(jiān)聽器,我們不做贅述,例如:

for(ApplicationListenerlistener:getApplicationListeners()){
if(listenerinstanceofSmartApplicationListener){
SmartApplicationListenersmartListener=(SmartApplicationListener)listener;
if(smartListener.supportsEventType(eventType)
&&smartListener.supportsSourceType(sourceType)){
invokeListener(smartListener,event);
}
}
elseif(supportsEvent(listener,eventType)){
invokeListener(listener,event);
}
}

基本上就是這些了。





審核編輯:劉清

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • 嵌入式
    +關(guān)注

    關(guān)注

    5046

    文章

    18817

    瀏覽量

    298555
  • Web服務(wù)器
    +關(guān)注

    關(guān)注

    0

    文章

    137

    瀏覽量

    24315
  • tomcat
    +關(guān)注

    關(guān)注

    0

    文章

    27

    瀏覽量

    4819
  • SpringBoot
    +關(guān)注

    關(guān)注

    0

    文章

    172

    瀏覽量

    145

原文標(biāo)題:三分鐘了解 SpringBoot 的啟動流程

文章出處:【微信號:芋道源碼,微信公眾號:芋道源碼】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。

收藏 人收藏

    評論

    相關(guān)推薦

    SpringBoot嵌入式Web容器有哪幾種呢

    種類1.servlett web容器,2.reactive web容器spring-boot支持使用jetty、netty、tomcat、undertow作為
    發(fā)表于 12-16 06:51

    談一談Spring Boot嵌入式Web容器

    Spring Boot嵌入式Web容器Embedded Tomcatorg.springframework.boot.context.embedded.EmbeddedServletContainerCustomizerorg.
    發(fā)表于 12-16 08:16

    Spring Boot嵌入式Web容器原理是什么

    Spring Boot嵌入式Web容器原理Spring Boot的目標(biāo)是構(gòu)建“非常容易創(chuàng)建、獨立、產(chǎn)品級別的基于Spring的應(yīng)用”。這些應(yīng)用是“立即可運行的”。在這個過程中,完全沒有代碼生成
    發(fā)表于 12-16 07:57

    SpringBoot嵌入式Servlet容器啟動原理是什么

    SpringBoot嵌入式Servlet容器啟動原理思維導(dǎo)圖
    發(fā)表于 12-20 07:26

    嵌入式Servlet容器啟動原理是什么

    一、SpringBoot應(yīng)用啟動運行run方法二、run方法調(diào)用了refreshContext(context);SpringBoot刷新IOC容器【創(chuàng)建IOC容器對象,并初始化
    發(fā)表于 12-20 07:54

    SpringBoot應(yīng)用啟動運行run方法

    什么時候創(chuàng)建嵌入式的Servlet容器工廠?什么時候獲取嵌入式的Servlet容器并啟動Tomc
    發(fā)表于 12-20 06:16

    嵌入式Servlet容器自動配置原理是什么

    這一節(jié)課我們來說一說嵌入式Servlet容器自動配置原理前面我們都知道怎么去配置容器參數(shù),切換容器,但是我們不知道springboot自動配
    發(fā)表于 12-20 06:29

    如何配置嵌入式的Servlet容器?

    以前的web應(yīng)用開發(fā)我們采取的方式是項目完成后打包成war包,然后配置tomcat啟動運行項目,而Spring Boot默認(rèn)使用的是嵌入式的tomcat,那我們需要如何配置嵌入式的Servlet
    發(fā)表于 12-20 07:18

    SpringBoot配置嵌入式Servlet

    SpringBoot配置嵌入式Servlet容器定制和修改Servlet容器相關(guān)配置全局配置文件編寫WebServerFactoryCustomizer注冊Servlet三大組件注冊S
    發(fā)表于 12-20 06:19

    什么時候獲取嵌入式的Servlet容器并啟動Tomcat

    什么時候創(chuàng)建嵌入式的Servlet容器工廠?什么時候獲取嵌入式的Servlet容器并啟動Tomc
    發(fā)表于 12-20 06:11

    如何對嵌入式Servlet容器進行配置呢

    一、配置嵌入式Servlet容器SpringBoot默認(rèn)使用Tomcat作為嵌入式的Servlet容器;1、定制和修改Servlet
    發(fā)表于 12-21 06:09

    嵌入式Servlet容器啟動原理

    SpringBoot源碼學(xué)習(xí)系列之嵌入式Servlet容器啟動原理SpringBoot的自動配置就是SpringBoot的精髓所在,對于
    發(fā)表于 12-22 07:23

    如何配置嵌入式Servlet容器

    springboot默認(rèn)使用的是嵌入式的servlet容器(tomcat):定制和修改servlet容器的相關(guān)配置1)修改和server有關(guān)的配置(ServerProperties類
    發(fā)表于 12-24 06:56

    嵌入式引導(dǎo)加載技術(shù)

    嵌入式引導(dǎo)加載技術(shù)
    發(fā)表于 02-08 01:37 ?10次下載

    嵌入式Servlet容器

    配置嵌入式Servlet容器##Spring Boot里面內(nèi)置了嵌入式的Servlet容器(tomcat)點擊pom.xml->右鍵->Diagrams->show
    發(fā)表于 10-20 17:51 ?3次下載
    <b class='flag-5'>嵌入式</b>Servlet<b class='flag-5'>容器</b>