SpringBoot启动之SpringApplication初始化详解

 更新时间:2024年01月02日 08:32:20   作者:my_sky_  
这篇文章主要介绍了SpringBoot启动之SpringApplication初始化详解,首先初始化资源加载器,默认为null;断言判断主要资源类不能为null,否则报错,需要的朋友可以参考下

SpringApplication初始化

在这里插入图片描述

当启动SpringBoot应用后,经过两步,会进入到new SpringApplication(primarySources).run(args)。

1、primarySources参数

primarySources参数实际为Spring Boot应用上下文的Configuration Class,在后面扫描配置类时起作用。

2、SpringApplication初始化

SpringApplication对象的初始化

在这里插入图片描述

具体操作包括:

  • 首先初始化资源加载器,默认为null;断言判断主要资源类不能为null,否者报错。
  • 然后将主资源类primarySources存储到SpringApplication对象Set类型的primarySources属性中。
  • 推断当前 WEB 应用类型,一共有三种:NONE,SERVLET,REACTIVE;默认是SERVLET。
  • 加载Spring应用上下文初始化器:从"META-INF/spring.factories"文件中读取ApplicationContextInitializer类的实例名称集合,然后进行Set去重、利用反射实例化对象,最后按照Order排序后,赋值到SpringApplication的List类型的initializers属性上,一共7个。
  • 加载Spring应用事件监听器:从"META-INF/spring.factories"文件中读取ApplicationListener类的实例名称集合,然后进行Set去重、利用反射实例化对象,最后按照Order排序后,赋值到SpringApplication的List类型的listeners属性上,一共11个。
  • 推断主入口应用类:通过当前调用栈的解析,获取Main方法所在类,并赋值给SpringApplication的mainApplicationClass属性。

1)推断Web应用类型

this.webApplicationType = WebApplicationType.deduceFromClasspath();

这里推断的Web应用类型是默认的,我们还可以手动的再通过setWebApplicatioinType(WebApplicationType)方法进行调整。这里通过检查当前ClassLoader下基准Class的存在性来推断Web应用类型。

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应用类型为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`不存在,则当前应用为非Web引应用,即NONE。
		for (String className : SERVLET_INDICATOR_CLASSES) {
			if (!ClassUtils.isPresent(className, null)) {
				return WebApplicationType.NONE;
			}
		}
		// 3.当Spring WebFlux和Spring Web MVC同时存在时,Web应用依旧是SERVLET。
		return WebApplicationType.SERVLET;
	}
    ....
}

WEB 应用类型,一共有三种:NONE,SERVLET,REACTIVE。

deduceFromClasspath()方法利用ClassUtils.isPresent(String, ClassLoader)方法依次判断reactive.DispatcherHandler、ConfigurableWebApplicationContext、Servlet、servlet.DispatcherServlet的存在性组合情况,从而判断Web 引用类型,具体逻辑如下:

  • 如果DispatcherHandler存在,并且DispatcherServlet和ServletContainer不存在时,即:Spring Boot仅依赖WebFlux时,Web应用类型为REACTIVE;
  • 如果Servlet和ConfigurableWebApplicationContext不存在,则当前应用为非Web应用,即NONE。因为这两个API是Spring Web MVC必须的依赖。
  • 当Spring WebFlux和Spring Web MVC同时存在时,Web应用类型依旧是SERVLET。

2)加载Spring应用上下文初始化器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事件应用监听器ApplicationListener

setListeners((Collection) getSpringFactoriesInstances(ApplicationListener.class));

这个过程和加载Spring应用上下文初始化器ApplicationContextInitializer一样(区别在于这里不再直接从spring.factories文件中获取内容,而是走cache(MultiValueMap<String, String>)缓存)。

4)推断应用引导类

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;
}

该方法根据当前线程执行栈来判断其栈中哪个类包含main方法,然后将找到的类名通过反射返回Class对象。

至此,在SpringApplication构造过程中,SpringApplication属性primarySources、webApplicationType、initializers、listeners 和 mainApplicationClass都被初始化了。

到此这篇关于SpringBoot启动之SpringApplication初始化详解的文章就介绍到这了,更多相关SpringApplication初始化内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Monaco Editor实现sql和java代码提示实现示例

    Monaco Editor实现sql和java代码提示实现示例

    这篇文章主要为大家介绍了Monaco Editor代码提示sql和java实现示例,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-08-08
  • java实现系统多级文件夹复制

    java实现系统多级文件夹复制

    这篇文章主要为大家详细介绍了java实现系统多级文件夹复制,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-08-08
  • 关于Spring不同类型的注入方式 p-namespace,c-namespace

    关于Spring不同类型的注入方式 p-namespace,c-namespace

    这篇文章主要介绍了Spring不同类型的注入方式 p-namespace,c-namespace。具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-09-09
  • springboot项目整合druid数据库连接池的实现

    springboot项目整合druid数据库连接池的实现

    这篇文章主要介绍了springboot项目整合druid数据库连接池的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2021-04-04
  • spring结合struts的代码详解

    spring结合struts的代码详解

    这篇文章主要介绍了spring结合struts的代码详解,需要的朋友可以参考下
    2017-09-09
  • Java设计模式之状态模式

    Java设计模式之状态模式

    这篇文章介绍了Java设计模式之状态模式,文中通过示例代码介绍的非常详细。对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2022-10-10
  • SpringBoot实现埋点监控

    SpringBoot实现埋点监控

    本文主要介绍了SpringBoot实现埋点监控,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-01-01
  • java整数与byte数组的转换实现代码

    java整数与byte数组的转换实现代码

    这篇文章主要介绍了java整数与byte数组的转换实现代码的相关资料,需要的朋友可以参考下
    2017-07-07
  • 关于Spring框架中异常处理情况浅析

    关于Spring框架中异常处理情况浅析

    最近学习Spring时,认识到Spring异常处理的强大,这篇文章主要给大家介绍了关于Spring框架中异常处理情况的相关资料,通过示例代码介绍的非常详细,需要的朋友可以参考下
    2021-08-08
  • 解决SpringMVC接收不到ajaxPOST参数的问题

    解决SpringMVC接收不到ajaxPOST参数的问题

    今天小编就为大家分享一篇解决SpringMVC接收不到ajaxPOST参数的问题,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2018-08-08

最新评论