SpringBoot启动原理——SpringApplication《课时十一》

简介: SpringBoot启动原理——SpringApplication《课时十一》

关于SpringApplication类的作用有两个结论

推荐:画流程图的软件 下载Boardmix协同设计工具 - Boardmix官网

结论一:SpringApplication类的作用

  • 1.推断应用的类型是普通的项目还是Web项目
  • 2.查找并加载所有可用初始化器,设 置到initializers属性中.
  • 3 找出所有的应用程序监听器,设置到listeners属性中
  • 4.推断并设置main方法的定义类,找到运行的主类

结论二:SpringApplication类的作用

  • 1.获取启动类
  • 2.获取web应用类型
  • 3.读取了对外扩展的Applcation Contextintializer ,ApplicationListener
  • 4.根据main推算出所在的类就是去初始化了一些信息


官网介绍:

Spring Boot Features 找到1.7的介绍

官网介绍的SpringBoot启动原理该如何启动 翻译在下面的文章中


当您的应用程序运行时,应用程序事件按以下顺序发送:


ApplicationStartingEvent在运行开始时但在任何处理之前发送,除了侦听器和初始化程序的注册。


ApplicationEnvironmentPreparedEvent当Environment在上下文中使用的 已知但在创建上下文之前发送一个。


ApplicationContextInitializedEvent当ApplicationContext准备好并调用 ApplicationContextInitializers 但在加载任何 bean 定义之前发送一个。


ApplicationPreparedEvent在刷新开始之前但在加载 bean 定义之后发送一个。


AnApplicationStartedEvent在上下文刷新之后但在调用任何应用程序和命令行运行程序之前发送。


AvailabilityChangeEvent在 with 之后立即发送一个LivenessState.CORRECT,表明应用程序被认为是活动的。


ApplicationReadyEvent在调用任何应用程序和命令行运行程序后发送一个。


AvailabilityChangeEvent在 with 之后立即发送一个ReadinessState.ACCEPTING_TRAFFIC,表示应用程序已准备好为请求提供服务。


如果ApplicationFailedEvent启动时出现异常,则发送一个。

这篇文章在解释SpringApplication类的作用:

package com.spring.springboot0907web;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
//深入源码 解析SpringBoot自动配置原理
@SpringBootApplication
public class Springboot0907WebApplication {
    public static void main(String[] args) {
        //解析SpringBoot启动原理
        SpringApplication.run(Springboot0907WebApplication.class, args);
    }
}

关注点在上面的代码中

 

按住Ctrl键:点击进入Run方法  

 

如图所示进入上面的页面

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

使用自定义的Spring Application 进行启动程序

按住Ctrl键:点击进入SpringApplication 进入下面的页面信息

 

按住 ctrl 进入 this


解析下面的源码 来得出我们自己的结论:

 public SpringApplication(ResourceLoader resourceLoader, Class<?>... primarySources) {
        this.sources = new LinkedHashSet();
        this.bannerMode = Mode.CONSOLE;
        this.logStartupInfo = true;
        this.addCommandLineProperties = true;
        this.addConversionService = true;
        this.headless = true;
        this.registerShutdownHook = true;
        this.additionalProfiles = Collections.emptySet();
        this.isCustomEnvironment = false;
        this.lazyInitialization = false;
        this.applicationContextFactory = ApplicationContextFactory.DEFAULT;
        this.applicationStartup = ApplicationStartup.DEFAULT;
        this.resourceLoader = resourceLoader;
        Assert.notNull(primarySources, "PrimarySources must not be null");
        this.primarySources = new LinkedHashSet(Arrays.asList(primarySources));
        this.webApplicationType = WebApplicationType.deduceFromClasspath();
        this.bootstrapRegistryInitializers = new ArrayList(this.getSpringFactoriesInstances(BootstrapRegistryInitializer.class));
        this.setInitializers(this.getSpringFactoriesInstances(ApplicationContextInitializer.class));
        this.setListeners(this.getSpringFactoriesInstances(ApplicationListener.class));
        this.mainApplicationClass = this.deduceMainApplicationClass();
    }

1 将启动类放在 primarySources 放在哪里干吗 接着往下看

this.primarySources = new LinkedHashSet(Arrays.asList(primarySources));

2 更据classpath下的类 推算当前的web应用类型(webflux,servlet)  NONE,SERVLET, REACTIVE;

  this.webApplicationType = WebApplicationType.deduceFromClasspath();
//点击进去 WebApplicationType 如下所示
public enum WebApplicationType {
    NONE,
    SERVLET,
    REACTIVE;
    private static final String[] SERVLET_INDICATOR_CLASSES = new String[]{"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 WebApplicationType() {
    }

3  又是去SpringFactories 中获取所有的key:org.springframework.context.ApplicationContextInitializer=\

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

4 又是去SpringFactories 中获取所有的key:org.springframework.context.ApplicationListener=\

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

5  根据main方法推算出mainApplicationClass

this.mainApplicationClass = this.deduceMainApplicationClass();

观察下面上面的图

SpringApplication依据源码分析:最终结论:

  1. 推断应用的类型是普通的项目还是Web项目。

查找并加载所有可用初始化器,设 置到initializers属性中。

找出所有的应用程序监听器,设置到listeners属性中。

更据classpath下的类 推算当前的web应用类型(webflux,servlet)  NONE,SERVLET, REACTIVE。

去SpringFactories 中获取所有的key:org.springframework.context.ApplicationContextInitializer=\

又是去SpringFactories 中获取所有的key:org.springframework.context.ApplicationListener=\

根据main推算出所在的类就是去初始化了一些信息,推断并设置main方法的定义类,找到运行的主类

理解图一

 

理解图二

相关文章
|
4月前
|
人工智能 Java 开发者
【Spring】原理解析:Spring Boot 自动配置
Spring Boot通过“约定优于配置”的设计理念,自动检测项目依赖并根据这些依赖自动装配相应的Bean,从而解放开发者从繁琐的配置工作中解脱出来,专注于业务逻辑实现。
1601 0
|
6月前
|
Java Spring 容器
SpringBoot自动配置的原理是什么?
Spring Boot自动配置核心在于@EnableAutoConfiguration注解,它通过@Import导入配置选择器,加载META-INF/spring.factories中定义的自动配置类。这些类根据@Conditional系列注解判断是否生效。但Spring Boot 3.0后已弃用spring.factories,改用新格式的.imports文件进行配置。
1052 0
|
3月前
|
JavaScript Java Maven
【SpringBoot(二)】带你认识Yaml配置文件类型、SpringMVC的资源访问路径 和 静态资源配置的原理!
SpringBoot专栏第二章,从本章开始正式进入SpringBoot的WEB阶段开发,本章先带你认识yaml配置文件和资源的路径配置原理,以方便在后面的文章中打下基础
379 3
|
3月前
|
XML Java 应用服务中间件
【SpringBoot(一)】Spring的认知、容器功能讲解与自动装配原理的入门,带你熟悉Springboot中基本的注解使用
SpringBoot专栏开篇第一章,讲述认识SpringBoot、Bean容器功能的讲解、自动装配原理的入门,还有其他常用的Springboot注解!如果想要了解SpringBoot,那么就进来看看吧!
479 2
|
6月前
|
前端开发 Java 数据库连接
SpringBoot参数校验底层原理和实操。深度历险、深度解析(图解+秒懂+史上最全)
SpringBoot参数校验底层原理和实操。深度历险、深度解析(图解+秒懂+史上最全)
SpringBoot参数校验底层原理和实操。深度历险、深度解析(图解+秒懂+史上最全)
|
10月前
|
Java Spring
SpringBoot自动配置原理
本文深入解析了SpringBoot的核心功能——自动配置,重点探讨了`org.springframework.boot.autoconfigure`及相关注解的工作机制。通过分析`@SpringBootApplication`、`@EnableAutoConfiguration`等注解,揭示了SpringBoot如何基于类路径和条件自动装配Bean
508 8
|
10月前
|
Java
SpringBoot自动装配的原理
在SpringBoot项目的启动引导类上都有一个注解@SpringBootApplication 这个注解是一个复合注解, 其中有三个注解构成 , 分别是 ● @SpringBootConfiguration : 是@Configuration的派生注解 , 标注当前类是一个SpringBoot的配置类 ● @ComponentScan : 开启组件扫描, 默认扫描的是当前启动引导了所在包以及子包 ● @EnableAutoConfiguration : 开启自动配置(自动配置核心注解) 2.在@EnableAutoConfiguration注解的内容使用@Import注解导入了一个AutoC
|
10月前
|
JavaScript 前端开发 Java
Idea启动SpringBoot程序报错:Veb server failed to start. Port 8082 was already in use;端口冲突的原理与解决方案
本文解决了Idea启动SpringBoot程序报错:Veb server failed to start. Port 8082 was already in use的问题,并通过介绍端口的使用原理和操作系统的端口管理机制,可以更有效地解决端口冲突问题,并确保Web服务器能够顺利启动和运行。 只有锻炼思维才能可持续地解决问题,只有思维才是真正值得学习和分享的核心要素。如果这篇博客能给您带来一点帮助,麻烦您点个赞支持一下,还可以收藏起来以备不时之需,有疑问和错误欢迎在评论区指出~
|
11月前
|
Java 数据库 开发者
详细介绍SpringBoot启动流程及配置类解析原理
通过对 Spring Boot 启动流程及配置类解析原理的深入分析,我们可以看到 Spring Boot 在启动时的灵活性和可扩展性。理解这些机制不仅有助于开发者更好地使用 Spring Boot 进行应用开发,还能够在面对问题时,迅速定位和解决问题。希望本文能为您在 Spring Boot 开发过程中提供有效的指导和帮助。
1334 12
|
9月前
|
安全 前端开发 Java
Spring Boot 项目中触发 Circular View Path 错误的原理与解决方案
在Spring Boot开发中,**Circular View Path**错误常因视图解析与Controller路径重名引发。当视图名称(如`login`)与请求路径相同,Spring MVC无法区分,导致无限循环调用。解决方法包括:1) 明确指定视图路径,避免重名;2) 将视图文件移至子目录;3) 确保Spring Security配置与Controller路径一致。通过合理设定视图和路径,可有效避免该问题,确保系统稳定运行。
617 0