SpringBoot啟動之SpringApplication初始化詳解
SpringApplication初始化
當啟動SpringBoot應(yīng)用后,經(jīng)過兩步,會進入到new SpringApplication(primarySources).run(args)。
1、primarySources參數(shù)
primarySources參數(shù)實際為Spring Boot應(yīng)用上下文的Configuration Class,在后面掃描配置類時起作用。
2、SpringApplication初始化
SpringApplication對象的初始化
具體操作包括:
- 首先初始化資源加載器,默認為null;斷言判斷主要資源類不能為null,否者報錯。
- 然后將主資源類primarySources存儲到SpringApplication對象Set類型的primarySources屬性中。
- 推斷當前 WEB 應(yīng)用類型,一共有三種:NONE,SERVLET,REACTIVE;默認是SERVLET。
- 加載Spring應(yīng)用上下文初始化器:從"META-INF/spring.factories"文件中讀取ApplicationContextInitializer類的實例名稱集合,然后進行Set去重、利用反射實例化對象,最后按照Order排序后,賦值到SpringApplication的List類型的initializers屬性上,一共7個。
- 加載Spring應(yīng)用事件監(jiān)聽器:從"META-INF/spring.factories"文件中讀取ApplicationListener類的實例名稱集合,然后進行Set去重、利用反射實例化對象,最后按照Order排序后,賦值到SpringApplication的List類型的listeners屬性上,一共11個。
- 推斷主入口應(yīng)用類:通過當前調(diào)用棧的解析,獲取Main方法所在類,并賦值給SpringApplication的mainApplicationClass屬性。
1)推斷Web應(yīng)用類型
this.webApplicationType = WebApplicationType.deduceFromClasspath();
這里推斷的Web應(yīng)用類型是默認的,我們還可以手動的再通過setWebApplicatioinType(WebApplicationType)方法進行調(diào)整。這里通過檢查當前ClassLoader下基準Class的存在性來推斷Web應(yīng)用類型。
public enum WebApplicationType { NONE, SERVLET, REACTIVE; private static final String[] SERVLET_INDICATOR_CLASSES = { "javax.servlet.Servlet", "org.springframework.web.context.ConfigurableWebApplicationContext" }; private static final String WEBMVC_INDICATOR_CLASS = "org.springframework.web.servlet.DispatcherServlet"; private static final String WEBFLUX_INDICATOR_CLASS = "org.springframework.web.reactive.DispatcherHandler"; private static final String JERSEY_INDICATOR_CLASS = "org.glassfish.jersey.servlet.ServletContainer"; .... static WebApplicationType deduceFromClasspath() { // 1. 如果`DispatcherHandler`存在,并且`DispatcherServlet`和`ServletContainer`不存在時,Web應(yīng)用類型為REACTIVE; if (ClassUtils.isPresent(WEBFLUX_INDICATOR_CLASS, null) && !ClassUtils.isPresent(WEBMVC_INDICATOR_CLASS, null) && !ClassUtils.isPresent(JERSEY_INDICATOR_CLASS, null)) { return WebApplicationType.REACTIVE; } // 2. 如果`Servlet`和`ConfigurableWebApplicationContext`不存在,則當前應(yīng)用為非Web引應(yīng)用,即NONE。 for (String className : SERVLET_INDICATOR_CLASSES) { if (!ClassUtils.isPresent(className, null)) { return WebApplicationType.NONE; } } // 3.當Spring WebFlux和Spring Web MVC同時存在時,Web應(yīng)用依舊是SERVLET。 return WebApplicationType.SERVLET; } .... }
WEB 應(yīng)用類型,一共有三種:NONE,SERVLET,REACTIVE。
deduceFromClasspath()方法利用ClassUtils.isPresent(String, ClassLoader)方法依次判斷reactive.DispatcherHandler、ConfigurableWebApplicationContext、Servlet、servlet.DispatcherServlet的存在性組合情況,從而判斷Web 引用類型,具體邏輯如下:
- 如果DispatcherHandler存在,并且DispatcherServlet和ServletContainer不存在時,即:Spring Boot僅依賴WebFlux時,Web應(yīng)用類型為REACTIVE;
- 如果Servlet和ConfigurableWebApplicationContext不存在,則當前應(yīng)用為非Web應(yīng)用,即NONE。因為這兩個API是Spring Web MVC必須的依賴。
- 當Spring WebFlux和Spring Web MVC同時存在時,Web應(yīng)用類型依舊是SERVLET。
2)加載Spring應(yīng)用上下文初始化器ApplicationContextInitializer
setInitializers((Collection) getSpringFactoriesInstances(ApplicationContextInitializer.class));
這個過程包括兩個動作:
1. getSpringFactoriesInstances(ApplicationContextInitializer.class)從"META-INF/spring.factories"文件中讀取ApplicationContextInitializer類的實例名稱集合,然后進行Set去重、利用反射實例化對象,最后按照Order排序。
2. setInitializers(Collection)將Collection賦值到SpringApplication的List類型的initializers屬性上,一共7個。
3)加載Spring事件應(yīng)用監(jiān)聽器ApplicationListener
setListeners((Collection) getSpringFactoriesInstances(ApplicationListener.class));
這個過程和加載Spring應(yīng)用上下文初始化器ApplicationContextInitializer一樣(區(qū)別在于這里不再直接從spring.factories文件中獲取內(nèi)容,而是走cache(MultiValueMap<String, String>)緩存)。
4)推斷應(yīng)用引導(dǎo)類
this.mainApplicationClass = deduceMainApplicationClass();
private Class<?> deduceMainApplicationClass() { try { StackTraceElement[] stackTrace = new RuntimeException().getStackTrace(); for (StackTraceElement stackTraceElement : stackTrace) { if ("main".equals(stackTraceElement.getMethodName())) { return Class.forName(stackTraceElement.getClassName()); } } } catch (ClassNotFoundException ex) { } return null; }
該方法根據(jù)當前線程執(zhí)行棧來判斷其棧中哪個類包含main方法,然后將找到的類名通過反射返回Class對象。
至此,在SpringApplication構(gòu)造過程中,SpringApplication屬性primarySources、webApplicationType、initializers、listeners 和 mainApplicationClass都被初始化了。
到此這篇關(guān)于SpringBoot啟動之SpringApplication初始化詳解的文章就介紹到這了,更多相關(guān)SpringApplication初始化內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Monaco?Editor實現(xiàn)sql和java代碼提示實現(xiàn)示例
這篇文章主要為大家介紹了Monaco?Editor代碼提示sql和java實現(xiàn)示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2022-08-08java實現(xiàn)系統(tǒng)多級文件夾復(fù)制
這篇文章主要為大家詳細介紹了java實現(xiàn)系統(tǒng)多級文件夾復(fù)制,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下2021-08-08關(guān)于Spring不同類型的注入方式 p-namespace,c-namespace
這篇文章主要介紹了Spring不同類型的注入方式 p-namespace,c-namespace。具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-09-09springboot項目整合druid數(shù)據(jù)庫連接池的實現(xiàn)
這篇文章主要介紹了springboot項目整合druid數(shù)據(jù)庫連接池的實現(xiàn),文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2021-04-04java整數(shù)與byte數(shù)組的轉(zhuǎn)換實現(xiàn)代碼
這篇文章主要介紹了java整數(shù)與byte數(shù)組的轉(zhuǎn)換實現(xiàn)代碼的相關(guān)資料,需要的朋友可以參考下2017-07-07解決SpringMVC接收不到ajaxPOST參數(shù)的問題
今天小編就為大家分享一篇解決SpringMVC接收不到ajaxPOST參數(shù)的問題,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2018-08-08