返回顶部
首页 > 资讯 > 精选 >SpringBoot启动原理是什么
  • 439
分享到

SpringBoot启动原理是什么

2023-07-05 21:07:49 439人浏览 安东尼
摘要

今天小编给大家分享一下SpringBoot启动原理是什么的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。入口版本: 2.1.8

今天小编给大家分享一下SpringBoot启动原理是什么的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。

入口

版本: 2.1.8.RELEASE

启动代码:

@springBootApplcationpublic static void main(String[] args) {    SpringApplication.run(BlogAdminApplication.class, args);    System.out.println("======== admin start success... ==========");}

这里传入了两个参数,BlogAdminApplication当前类和args参数

我们点击进入run方法查看

public static ConfigurableApplicationContext run(Class<?> primarySource, String... args) {    return run(new Class[]{primarySource}, args);}

这里是将我们写的启动类传入到了Class[]数组中,这步就是个单纯的参数转换。

探讨primarySource参数

那么问题是primarySource能接受的类型是啥样的,是不是什么类都可以接受,带着这个疑问,我们做一个测试,把这个参数给换成一个别的类呢,ManagerController类是一个我写的接口类

@SpringBootApplicationpublic class BlogProjectApplication {   public static void main(String[] args) {       SpringApplication.run(ManagerController.class, args);       System.out.println("======== admin start success... ==========");   }}

控制台打印

org.springframework.context.ApplicationContextException: Unable to start WEB server; nested exception is org.springframework.context.ApplicationContextException: Unable to start ServletWebServerApplicationContext due to missing ServletWebServerFactory bean.

提示不能启动服务,提示缺少了ServletWebServerFactory bean 点进这个类看下

@FunctionalInterfacepublic interface ServletWebServerFactory {    WebServer getWebServer(ServletContextInitializer... initializers);}

他被FunctionalInterface标注了,是一个函数式接口,只有一个getWebServer方法,用来获取webServer的 看下他的实现类,

SpringBoot启动原理是什么

这不就是提示我们缺少启动的服务容器么,说的直白点,我的理解就是他缺少可以运行的容器,我们知道,没有使用springboot项目之前,我们的项目都是跑在Tomcat容器上的,当然也有使用Jetty容器的。再者,我们知道SpringBoot是对tomcat进行了内置。而SpringBoot不仅仅是只有内置了tomcat,而且还内置了好多的东西,比如我们经常使用的MQredis等等一系列的东西,这个我们可以在spring.factories配置文件中看到,这个文件位于如下位置

SpringBoot启动原理是什么

大概内容有下,篇幅有限,就不一一列举了。

省略。。。org.springframework.boot.autoconfigure.EnableAutoConfiguration=\org.springframework.boot.autoconfigure.admin.SpringApplicationAdminJmxAutoConfiguration,\org.springframework.boot.autoconfigure.aop.AopAutoConfiguration,\org.springframework.boot.autoconfigure.amqp.RabbitAutoConfiguration,\org.springframework.boot.autoconfigure.batch.BatchAutoConfiguration,\org.springframework.boot.autoconfigure.cache.CacheAutoConfiguration,\省略。。。

那么回过头来,我们再看下这个问题,这些个类是如何被加载进来的,我们知道SpingBoot有个注解是开启自动注解的@EnableAutoConfiguration,他就干这个事情的。他能够激活SpringBoot内建和自定义组件的自动装配特性。

那么,知道了这些,我们把这个之前修改后的类给改造一下,加上注解@EnableAutoConfiguration,看下执行效果。

@RestController@RequestMapping("project/manager")@EnableAutoConfigurationpublic class ManagerController extends AbstractController {

运行如下

SpringBoot启动原理是什么

从打印信息就能知道,服务器有了,只不过下面报错,提示找不到bean,那这不就简单了么,他是不是就是没有扫描到我们的包么,这里就其实可以在配置扫描包的注解继续测试,我就懒的不测试了,直接去看@SpringBootApplication注解

@Target({ElementType.TYPE})@Retention(RetentionPolicy.RUNTIME)@Documented@Inherited@SpringBootConfiguration@EnableAutoConfiguration//@ComponentScan(    excludeFilters = {@Filter(    type = FilterType.CUSTOM,    classes = {TypeExcludeFilter.class}), @Filter(    type = FilterType.CUSTOM,    classes = {AutoConfigurationExcludeFilter.class})})public @interface SpringBootApplication {

@SpringBootApplication这个注解其实就是一个组合注解,里面包含了

  • 元注解:用来标注他是一个注解的,jdk自带的

  • @SpringBootConfiguration:集成自@Configuration,表示是一个配置类

  • @EnableAutoConfiguration:激活SpringBoot内建和自定义组件的自动装配特性

  • @ComponentScan:扫描注解,添加了排除参数,指定排除了一些类

通过这里的实验,我们可以得出结论:

primarySource参数能接收的类是一个配置类,同时要把符合扫描规则的类装配到spring容器中,并且对SpringBoot内置的一些类进行自动扫描到,而这里的@SpringBootApplication注解就是把这些特性都整合到了一起,作为了一个引导类而已。那么说白了,primarySource他接受的其实就是一个配置类。

关于注解详细知识的话,这里就聊这么多了,后面再详细聊。

args参数

args是Java命令行参数,我们在DOS中执行Java程序的时候使用“java 文件名 args参数”。args这个数组可以接收到这些参数。这个是个基础常识了。

以下我们将继续跟踪源码进行分析

我们继续追run()方法

public static ConfigurableApplicationContext run(Class&lt;?&gt;[] primarySources, String[] args) {    return (new SpringApplication(primarySources)).run(args);}

这个方法干了两个事情:

new SpringApplication()来创建对象

通过创建后的对象,调用对象里面的run()方法

以下我们将从这两个地方进行分析,本篇就先研究第一个

创建对象

我们先看下他是怎么创建对象的,创建了哪些对象,

public SpringApplication(Class<?>... primarySources) {    this((ResourceLoader)null, primarySources);}public SpringApplication(ResourceLoader resourceLoader, Class<?>... primarySources) {    //资源加载器   this.resourceLoader = resourceLoader;   //断言   Assert.notNull(primarySources, "PrimarySources must not be null");   //对primarySources进行存储到LinkedHashSet   this.primarySources = new LinkedHashSet<>(Arrays.asList(primarySources));   //1、推断web应用类别   this.webApplicationType = WebApplicationType.deduceFromClasspath();   //2、加载Spring应用上下文初始化   setInitializers((Collection) getSpringFactoriesInstances(ApplicationContextInitializer.class));   //3、加载Spring应用事件监听器   setListeners((Collection) getSpringFactoriesInstances(ApplicationListener.class));   //4、推断应用引导类   this.mainApplicationClass = deduceMainApplicationClass();}

下面研究下主要流程部分

1、推断web应用类别

推断web应用类型属于SpringBoot应用web类型的初始化过程。而该类型也可在SpringApplication构造后,run方法执行之前,通过setWebApplicationType(WebApplicationType webApplicationType)方法进行调整。

在推断Web应用类型的过程中,由于当前Spring应用上下文尚未准备(可在代码执行顺序中看到),所以实现采用的是检查检查当前ClassLoader下基准Class的存在性判断。

上源码

this.webApplicationType = WebApplicationType.deduceFromClasspath();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";private static final String SERVLET_APPLICATION_CONTEXT_CLASS = "org.springframework.web.context.WebApplicationContext";private static final String REACTIVE_APPLICATION_CONTEXT_CLASS = "org.springframework.boot.web.reactive.context.ReactiveWebApplicationContext";static WebApplicationType deduceFromClasspath() {//当DispatcherHandler存在,且DispatcherServlet、ServletContainer两个不存在时;换言之,SpringBoot仅依赖WebFlux存在时,此时的应用类型为REACTIVE   if (ClassUtils.isPresent(WEBFLUX_INDICATOR_CLASS, null) && !ClassUtils.isPresent(WEBMVC_INDICATOR_CLASS, null)         && !ClassUtils.isPresent(JERSEY_INDICATOR_CLASS, null)) {      return WebApplicationType.REACTIVE;   }   //当Servlet和ConfigurableWebApplicationContext均不存在时,当前应用为非Web应用,即WebApplicationType.NONE,因为这些api均是Spring Web MVC必须的依赖   for (String className : SERVLET_INDICATOR_CLASSES) {      if (!ClassUtils.isPresent(className, null)) {         return WebApplicationType.NONE;      }   }   //当WebFlux和Spring Web MVC同时存在时,Web应用类型同样是Servlet Web,即WebApplicationType.SERVLET   return WebApplicationType.SERVLET;}

2、加载Spring应用上下文初始化

setInitializers((Collection) getSpringFactoriesInstances(ApplicationContextInitializer.class));

此过程包含两个动作,依次为getSpringFactoriesInstances(ApplicationContextInitializer.class)和setInitializers方法。 先看第一个过程

private <T> Collection<T> getSpringFactoriesInstances(Class<T> type) {   return getSpringFactoriesInstances(type, new Class<?>[] {});}private <T> Collection<T> getSpringFactoriesInstances(Class<T> type, Class<?>[] parameterTypes, Object... args) {   //获取类加载器   ClassLoader classLoader = getClassLoader();   // Use names and ensure unique to protect against duplicates   //加载了META-INF/spring.factories资源中配置的ApplicationContextInitializer实现类名单。   Set<String> names = new LinkedHashSet<>(SpringFactoriesLoader.loadFactoryNames(type, classLoader));   //初始化   List<T> instances = createSpringFactoriesInstances(type, parameterTypes, classLoader, args, names);   AnnotationAwareOrderComparator.sort(instances);   return instances;}

此处使用了Spring工厂加载机制方法SpringFactoriesLoader.loadFactoryNames(type, classLoader)。加载了META-INF/spring.factories资源中配置的ApplicationContextInitializer实现类名单。

SpringBoot启动原理是什么

加载完成后使用createSpringFactoriesInstances方法对其进行初始化。

private <T> List<T> createSpringFactoriesInstances(Class<T> type, Class<?>[] parameterTypes,      ClassLoader classLoader, Object[] args, Set<String> names) {   List<T> instances = new ArrayList<>(names.size());   for (String name : names) {      try {      //从类加载器中获取指定类         Class<?> instanceClass = ClassUtils.forName(name, classLoader);         //判断instanceClass是不是type的子类         Assert.isAssignable(type, instanceClass);         //根据以上获取的类名创建类的实例         Constructor<?> constructor = instanceClass.getDeclaredConstructor(parameterTypes);         //排序         T instance = (T) BeanUtils.instantiateClass(constructor, args);         instances.add(instance);      }      catch (Throwable ex) {         throw new IllegalArgumentException("Cannot instantiate " + type + " : " + name, ex);      }   }   return instances;}

3、加载Spring应用事件监听器

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

此过程与2、加载上下文初始化基本类似。 只不过初始化的对象类型变成了ApplicationListener.class,setListeners方法也只是赋值而已

public void setListeners(Collection<? extends ApplicationListener<?>> listeners) {   this.listeners = new ArrayList<>();   this.listeners.addAll(listeners);}

SpringBoot启动原理是什么

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) {     // Swallow and continue  }  return null;}

这里使用到了new RuntimeException().getStackTrace()来获取堆栈信息,找到调用执行的main方法,从而确定他的类。
这里有个疑问:他不是传了primarySources数组,里面包含了类名么,怎么还用堆栈的方式去获取,此外,这里的堆栈获取也只能获取一个调用的主main方法,他为啥还要传一个Class数组呢?

具体咋获取的,可以追下源码,一直跟踪他的父类Throwable,找到如下代码

public synchronized Throwable fillInStackTrace() {    if (stackTrace != null ||        backtrace != null  ) {        fillInStackTrace(0);        stackTrace = UNASSIGNED_STACK;    }    return this;}private native Throwable fillInStackTrace(int dummy);

这里最后调用了native本地方法,去爬取线程堆栈信息,为运行时栈做一份快照。

SpringBoot启动原理是什么

通过这个图片,可以看到整个方法的调用链,从下往上看哦

以上就是“SpringBoot启动原理是什么”这篇文章的所有内容,感谢各位的阅读!相信大家阅读完这篇文章都有很大的收获,小编每天都会为大家更新不同的知识,如果还想学习更多的知识,请关注编程网精选频道。

--结束END--

本文标题: SpringBoot启动原理是什么

本文链接: https://lsjlt.com/news/355136.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • SpringBoot启动原理是什么
    今天小编给大家分享一下SpringBoot启动原理是什么的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。入口版本: 2.1.8...
    99+
    2023-07-05
  • springboot启动的原理是什么
    Spring Boot的启动原理可以分为以下几个步骤: 加载依赖:Spring Boot会根据项目的配置文件(如pom.xml)...
    99+
    2023-10-26
    springboot
  • springmvc启动原理是什么
    Spring MVC的启动原理如下:1. 当应用程序启动时,Servlet容器(如Tomcat)会加载web.xml文件,并解析其中...
    99+
    2023-09-21
    springmvc
  • SpringBoot嵌入式web容器的启动原理是什么
    这篇文章将为大家详细讲解有关SpringBoot嵌入式web容器的启动原理是什么,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。SpringBoot应用启动run方法SpringApplication.ja...
    99+
    2023-06-25
  • springboot自动配置原理是什么
    Spring Boot的自动配置原理是基于条件注解和条件判断的。Spring Boot通过在配置类上使用@EnableAutoCon...
    99+
    2023-08-23
    springboot
  • Spring Boot启动的原理是什么
    本文小编为大家详细介绍“Spring Boot启动的原理是什么”,内容详细,步骤清晰,细节处理妥当,希望这篇“Spring Boot启动的原理是什么”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起...
    99+
    2024-04-02
  • java SpringBoot自动装配原理是什么
    这篇文章主要介绍“java SpringBoot自动装配原理是什么”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“java SpringBoot自动装配原理是什么”文章能帮助大家解决问题。summar...
    99+
    2023-07-06
  • springboot启动流程是什么
    Spring Boot 启动流程如下:1. 初始化应用程序上下文:Spring Boot 应用程序启动时,首先会创建一个 Sprin...
    99+
    2023-05-17
    springboot启动流程 springboot
  • springboot启动时没有日志的原因是什么
    这篇文章主要介绍“springboot启动时没有日志的原因是什么”,在日常操作中,相信很多人在springboot启动时没有日志的原因是什么问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”springboot启...
    99+
    2023-06-20
  • Springboot启动原理和自动配置原理解析
    目录启动原理SpringApplication1、初始化2、调用run方法自动配置原理放本地文件夹都快吃土了,准备清理文件夹,关于Springboot的! 启动原理 @SpringB...
    99+
    2023-05-17
    Springboot启动原理和自动配置 Springboot自动配置 Springboot启动
  • SpringBoot启动原理深入解析
    目录一、SpringBootApplication 背后的秘密1、@Configuration2、@ComponentScan3、@EnableAutoConfiguration二、...
    99+
    2023-05-14
    springboot启动流程 SpringBoot的启动类 springboot启动原理
  • Springboot启动原理详细讲解
    主启动类方法: @SpringBootApplication public class MyJavaTestApplication { public static void ...
    99+
    2024-04-02
  • SpringBoot内置tomcat启动原理详解
    前言 不得不说SpringBoot的开发者是在为大众程序猿谋福利,把大家都惯成了懒汉,xml不配置了,连tomcat也懒的配置了,典型的一键启动系统,那么tomcat在spring...
    99+
    2024-04-02
  • SpringBoot应用jar包启动原理详解
    目录1、maven打包2、Jar包目录结构3、可执行Jar(JarLauncher)4、WarLauncher5、总结1、maven打包 Spring Boot项目的pom.xml文...
    99+
    2024-04-02
  • SpringBoot中WEB的启动流程是什么
    这篇文章主要介绍“SpringBoot中WEB的启动流程是什么”,在日常操作中,相信很多人在SpringBoot中WEB的启动流程是什么问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”SpringBoot中WE...
    99+
    2023-06-29
  • 浅谈什么是SpringBoot异常处理自动配置的原理
    异常处理自动配置 ErrorMvcAutoConfiguration自动配置类自动配置了处理规则,给容器中注册了多种组件 errorAttributes组件,类型为DefaultEr...
    99+
    2024-04-02
  • idea springboot启动配置的方法是什么
    Spring Boot的启动配置有两种方法:1. 使用application.properties文件:可以在src/main/re...
    99+
    2023-09-21
    idea springboot
  • springboot启动类的三大注解是什么
    Spring Boot启动类通常使用三个注解:@SpringBootApplication、@EnableAutoConfigura...
    99+
    2023-08-31
    springboot
  • mysql启动失败的原因是什么
    本篇内容主要讲解“mysql启动失败的原因是什么”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“mysql启动失败的原因是什么”吧!一、一站式解决 1. 问题分析定位# 找到M...
    99+
    2023-06-20
  • tomcat启动不起来是什么原因
    本篇内容主要讲解“tomcat启动不起来是什么原因”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“tomcat启动不起来是什么原因”吧!目录现象: 端口占用:文件拼写错误:现象:  tomcat安...
    99+
    2023-06-20
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作