欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

SpringBoot啟動之SpringApplication初始化詳解

 更新時間:2024年01月02日 08:32:20   作者:my_sky_  
這篇文章主要介紹了SpringBoot啟動之SpringApplication初始化詳解,首先初始化資源加載器,默認為null;斷言判斷主要資源類不能為null,否則報錯,需要的朋友可以參考下

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)文章

最新評論