一、前言

在spring时代配置文件的加载都是通过web.xml配置加载的(Servlet3.0之前),可能配置方式有所不同,但是大多数都是通过指定路径的文件名的形式去告诉spring该加载哪个文件;

<context-param><param-name>contextConfigLocation</param-name><param-value>/WEB-INF/application*.xml</param-value>
</context-param>

而到了springboot时代,我们发现原来熟悉的web.xml已不复存在,但是springboot却依然可以找到默认的配置文件(application.yml),那它是如何实现的呢?今天我们就一起来探究一下springboot自动加载配置文件的机制!

看完本篇文章你将了解到:

  1. springboot什么时候加载配置文件
  2. springboot通过哪个类加载配置文件
  3. springboot自动加载配置文件流程
  4. 激活文件优先级
  5. 文件加载路径优先级
  6. 文件后缀优先级

二、提出猜想

我们知道在使用springboot中我们只要在resources下面新建一个application.yml文件他就会自动加载,那是不是springboot默认在哪里配置了这个路径和文件名?

三、验证猜想

为了证实我们的猜想,我们可以通过查看springboot项目源码,跟着debug一步一步走;
这里我使用的是springboot2.0版本,2.0与1.5版本比较启动的大体流程是一样的,只不过在一些实现中有所差异;

1.启动流程

要知道springboot如何加载配置文件,就需要了解它的启动流程:

我们从main方法进入,大概的调用流程如下:

DemoApplication.main->SpringApplication.run->new SpringApplication().run


其实启动的主要过程都在new SpringApplication().run();

  • new SpringApplication():创建SpringApplication实例,负责加载配置一些基本的环境变量、资源、构造器、监听器
  • run():负责springboot整个启动过程,包括加载创建环境、打印banner、配置文件、配置应用上下文,加载bean等等sb整个生命周期几乎都在run方法中;

今天我们的主题是sb如何加载配置文件,所以着重讲解加载配置文件和之前的操作原理和源码,其他的功能以后有机会再和大家一起研究,下面我们来看看new SpringApplication()做了什么操作;

2.创建SpringApplication实例

/*** 创建一个SpringApplication实体,应用程序上下文将从指定的主源文档加载bean以获取详细信息,* 这个实例可以在调用之前自定义* @param resourceLoader* @param primarySources*/
@SuppressWarnings({ "unchecked", "rawtypes" })
public SpringApplication(ResourceLoader resourceLoader, Class<?>... primarySources) {//使用的资源加载器this.resourceLoader = resourceLoader;//主要的bean资源 primarySources【在这里是启动类所在的.class】,不能为null,如果为null,抛异常Assert.notNull(primarySources, "PrimarySources must not be null");//启动类的实例数组转化成list,放在LinkedHashSet集合中this.primarySources = new LinkedHashSet<>(Arrays.asList(primarySources));/*** 创建应用类型,不同应用程序类型,创建不同的环境* springboot1.5 只有两种类型:web环境和非web环境* springboot2.0 有三种应用类型:WebApplicationType* NONE:不需要再web容器的环境下运行,也就是普通的工程* SERVLET:基于servlet的Web项目* REACTIVE:响应式web应用reactive web Spring5版本的新特性*/this.webApplicationType = WebApplicationType.deduceFromClasspath();/*** 每一个initailizer都是一个实现了ApplicationContextInitializer接口的实例。* ApplicationContextInitializer是Spring IOC容器中提供的一个接口: void initialize(C applicationContext);* 这个方法它会在ConfigurableApplicationContext的refresh()方法调用之前被调用(prepareContext方法中调用),* 做一些容器的初始化工作。*/setInitializers((Collection) getSpringFactoriesInstances(ApplicationContextInitializer.class));/*** Springboot整个生命周期在完成一个阶段的时候都会通过事件推送器(EventPublishingRunListener)产生一个事件(ApplicationEvent),* 然后再遍历每个监听器(ApplicationListener)以匹配事件对象,这是一种典型的观察者设计模式的实现* 具体事件推送原理请看:sb事件推送机制图*/setListeners((Collection) getSpringFactoriesInstances(ApplicationListener.class));// 指定main函数启动所在的类,即启动类BootApplication.classthis.mainApplicationClass = deduceMainApplicationClass();
}

我们来大概的看下ApplicationListener的一些实现类以及他们具体的功能简介

这些监听器的实现类都是在spring.factories文件中配置好的,代码中通过getSpringFactoriesInstances方法获取,这种机制叫做SPI机制:通过本地的注册发现获取到具体的实现类,轻松可插拔。

SpringBoot默认情况下提供了两个spring.factories文件,分别是:

spring-boot-2.0.2.RELEASE.jar
spring-boot-autoconfigure-2.0.2.RELEASE.jar

概括来说在创建SpringApplication实例的时候,sb会加载一些初始化和启动的参数与类,如同跑步比赛时的等待发令枪的阶段;

3.run方法

(1)、事件推送原理

SB启动过程中分多个阶段或者说是多个步骤,每完成一步就会产生一个事件,并调用对应事件的监听器,这是一种标准的观察者模式,这在启动的过程中有很好的扩展性,下面我们来看看sb的事件推送原理:
SpringBoot事件推送原理图:

(2)、run方法整体流程简述

/*** 运行应用程序,创建并刷新一个新的应用程序上下文** @param args* @return*/
public ConfigurableApplicationContext run(String... args) {/***  StopWatch: 简单的秒表,允许定时的一些任务,公开每个指定任务的总运行时间和运行时间。*  这个对象的设计不是线程安全的,没有使用同步。SpringApplication是在单线程环境下,使用安全。*/StopWatch stopWatch = new StopWatch();// 设置当前启动的时间为系统时间startTimeMillis = System.currentTimeMillis();stopWatch.start();// 创建一个应用上下文引用ConfigurableApplicationContext context = null;// 异常收集,报告启动异常Collection<SpringBootExceptionReporter> exceptionReporters = new ArrayList<>();/*** 系统设置headless模式(一种缺乏显示设备、键盘或鼠标的环境下,比如服务器),* 通过属性:java.awt.headless=true控制*/configureHeadlessProperty();/** 获取事件推送监器,负责产生事件,并调用支某类持事件的监听器* 事件推送原理看上面的事件推送原理图*/SpringApplicationRunListeners listeners = getRunListeners(args);/*** 发布一个启动事件(ApplicationStartingEvent),通过上述方法调用支持此事件的监听器*/listeners.starting();try {// 提供对用于运行SpringApplication的参数的访问。取默认实现ApplicationArguments applicationArguments = new DefaultApplicationArguments(args);/*** 构建容器环境,这里加载配置文件*/ConfigurableEnvironment environment = prepareEnvironment(listeners, applicationArguments);// 对环境中一些bean忽略配置configureIgnoreBeanInfo(environment);// 日志控制台打印设置Banner printedBanner = printBanner(environment);// 创建容器context = createApplicationContext();exceptionReporters = getSpringFactoriesInstances(SpringBootExceptionReporter.class, new Class[] { ConfigurableApplicationContext.class }, context);/*** 准备应用程序上下文* 追踪源码prepareContext()进去我们可以发现容器准备阶段做了下面的事情:* 容器设置配置环境,并且监听容器,初始化容器,记录启动日志,* 将给定的singleton对象添加到此工厂的singleton缓存中。* 将bean加载到应用程序上下文中。*/prepareContext(context, environment, listeners, applicationArguments, printedBanner);/*** 刷新上下文* 1、同步刷新,对上下文的bean工厂包括子类的刷新准备使用,初始化此上下文的消息源,注册拦截bean的处理器,检查侦听器bean并注册它们,实例化所有剩余的(非延迟-init)单例。* 2、异步开启一个同步线程去时时监控容器是否被关闭,当关闭此应用程序上下文,销毁其bean工厂中的所有bean。* 。。。底层调refresh方法代码量较多*/refreshContext(context);afterRefresh(context, applicationArguments);// stopwatch 的作用就是记录启动消耗的时间,和开始启动的时间等信息记录下来stopWatch.stop();if (this.logStartupInfo) {new StartupInfoLogger(this.mainApplicationClass).logStarted(getApplicationLog(), stopWatch);}// 发布一个已启动的事件listeners.started(context);callRunners(context, applicationArguments);}catch (Throwable ex) {handleRunFailure(context, ex, exceptionReporters, listeners);throw new IllegalStateException(ex);}try {// 发布一个运行中的事件listeners.running(context);}catch (Throwable ex) {// 启动异常,里面会发布一个失败的事件handleRunFailure(context, ex, exceptionReporters, null);throw new IllegalStateException(ex);}return context;
}

(3)、构建容器环境

在:run方法中的ConfigurableEnvironment environment = prepareEnvironment(listeners, applicationArguments);是准备环境,里面会加载配置文件;

private ConfigurableEnvironment prepareEnvironment(SpringApplicationRunListeners listeners, ApplicationArguments applicationArguments) {// 创建一个配置环境,根据前面定义的应用类型定义不同的环境ConfigurableEnvironment environment = getOrCreateEnvironment();// 将配置参数设置到配置环境中configureEnvironment(environment, applicationArguments.getSourceArgs());/*** 发布一个环境装载成功的事件,并调用支持此事件的监听器* 这其中就有我们今天的主角:配置文件加载监听器(ConfigFileApplicationListener)*/listeners.environmentPrepared(environment);// 将配置环境绑定到应用程序bindToSpringApplication(environment);if (!this.isCustomEnvironment) {environment = new EnvironmentConverter(getClassLoader()).convertEnvironmentIfNecessary(environment, deduceEnvironmentClass());}ConfigurationPropertySources.attach(environment);return environment;
}

(4)、ConfigFileApplicationListener类介绍

sb就是通过ConfigFileApplicationListener 这个类来加载配置文件的,这个类同样是一个监听器,我们来看看他的继承类图:

再让我们来看看这个类具体都有哪些方法:

最后我们来看看这个类有哪些需要注意的字段:

(5)、ConfigFileApplicationListener类加载配置文件

我们从ConfigFileApplicationListener.onApplicationEvent开始,一直往下看方法链,发现最后是load方法去具体怎么加载配置文件的

激活配置文件与默认配置文件的优先级:
我们在使用中经常会根据不同的环境根据spring.profiles.active属性来定义不同的配置文件:

  • application-dev.properties
  • application-test.properties
  • application-prod.properties

但同时我们会创建一个默认的配置文件:application.properties,那自定义环境的配置文件与默认的配置文件的优先级是哪个高呢?

看图片我们可知他们加载的先后顺序(注意:后加载会覆盖前加载的文件):

  • application-xxx.properties
  • application.properties

配置文件路径的优先级:
我们从属性:DEFAULT_SEARCH_LOCATIONS = "classpath:/,classpath:/config/,file:./,file:./config/可以看出文件路径的先后顺序(注意:后加载的会覆盖先加载的):

  • classpath:/
  • classpath:/config/
  • file:./
  • file:./config/

配置文件的优先级:
我们从这个类中的字段:propertySourceLoaders可以看出有两个Loader,请各位看官看图:


我们从上面两张图中可以看出,每个Loader会加载两种后缀名的文件,加起来就是4种,又因为是数组类型,所以也会有先后顺序,所以加载配置文件的先后顺序就是(后加载覆盖先加载的):

  • properties
  • xml
  • yml
  • yaml

最后查找的具体路径:location + name + "-" + profile + "." + ext

这里我们介绍了三种优先级:

  1. active与默认优先级
  2. 文件路径优先级
  3. 文件后缀优先级
    未完待续。。。

四、提问

springboot学习遗留问题,
1.active和默认的谁覆盖谁
2.flter区别
3.多个配置文件如何覆盖

更多Java优质文章,请关注猪哥微信公众号:猪哥Java!

SpringBoot启动如何加载application.yml配置文件相关推荐

  1. Springboot默认加载application.yml原理

    Springboot默认加载application.yml原理以及扩展 SpringApplication.run(-)默认会加载classpath下的application.yml或applicat ...

  2. Spring Boot学习总结(13)——Spring Boot加载application.properties配置文件顺序规则

    SpringApplication会从以下路径加载所有的application.properties文件: 1.file:./config/(当前目录下的config文件夹) 2.file:./(当前 ...

  3. SpringBoot - application.yml配置文件中yes/no,on/off在代码中读取的值为true/false

    写在前面 在SpringBoot的项目中,当在配置文件中配置的值为yes/no或者on/off时,在SpringBoot内部解析时会将yes/no,on/off解析为true/false. 参数配置 ...

  4. Spring加载多个配置文件

    对于大多数的应用,从表现层的action,到持久层的DataSource,都被Spring 作为 bean 管理.如果这些bean 被配置在同一个文件中,阅读及维护该配置文件将是一件非 常有挑战的事情 ...

  5. Selenium3 Python WebDriver API源码探析(19)加载FireFox用户配置文件

    FireFox用户配置文件 Firefox 将用户个人信息(例如书签.密码.首选项.扩展.Cookie.证书等)保存在一系列文件中,它们被叫做用户配置文件,它们与 Firefox 的程序文件保存在不同 ...

  6. springboot多环境加载yml和logback配置

    大家好,我是烤鸭: 这是一篇关于springboot多环境加载yml和logback配置文件. 环境: 开发工具 idea(推荐)/eclipse(对yml支持不好) jdk  1.8 springb ...

  7. 配置文件加载优先级和外部配置文件加载||IDEA外部配置文件无法加载

    1,项目内部配置文件 spring boot 启动会扫描以下位置的 application.properties 或者 application.yml 文件作为 Spring boot 的默认配置文件 ...

  8. springboot mybatis 热加载mapper.xml文件(最简单)

    大家好,我是烤鸭: 今天介绍一下springboot mybatis 热加载mapper.xml文件. 本来不打算写的,看到网上比较流行的方式都比较麻烦,想着简化一下. 网上流行的版本. https: ...

  9. Tomcat原理系列之四:Tomat如何启动spring(加载web.xml)

    Tomcat原理系列之四:Tomat如何启动spring 熟悉的web.xml ContextLoaderListener Tomcat的初始化StandardContext.startInterna ...

最新文章

  1. MySQL如何选择数据类型
  2. MYSQL GROUP_CONCAT 用法
  3. 专科学会计还是计算机应用技术好,专科毕业想要学习会计专业,我专科是学计算机应用,但是毕业后家里让我学会计这门专业,具体是该怎么办呢...
  4. python 示例_带有示例的Python列表remove()方法
  5. linux 查找文件 mysql数据库_Linux下MySQL数据库目录多了好多文件
  6. 浏览器HTTP缓存机制 1
  7. 第15天android:使用sqlite
  8. linux自建git仓库
  9. CentOS6 图形界面(gnome)安装(转)
  10. LINUX SHELL判断文件、目录是否存在
  11. 基于vue.js仿淘宝收货地址,并设置默认地址
  12. php 开发工具 sublime,PHP日常开发工具-Sublime应用
  13. java后端开发(二):web开发历史解读
  14. 计算机应届生月薪大多是多少?
  15. SAP 订单结算方式
  16. 星巴克中国虎年新春限定产品上市
  17. 一文读懂IT行业都有哪些职位
  18. reference pics
  19. 基于“业务中台”构建的一些理解
  20. springboot启动报org.springframework.beans.factory.UnsatisfiedDependencyException: Error creating bean..

热门文章

  1. python库文档的错误_自己编程中遇到的Python错误和解决方法汇总整理
  2. 饥荒海难机器人怎么用_饥荒开发商又一款神作,难到吐血的生存游戏!
  3. 机器学习中的逻辑回归
  4. 10万奖金!探索图像盲降噪新方式,旷视2022 MegCup炼丹大赛等你来战
  5. 年末最大AI盛典!2020深度学习开发者峰会报名启动
  6. 从无监督构建词库看「最小熵原理」,套路是如何炼成的
  7. php oo,OO思想之PHP之三大特性
  8. 用python读取文档_python读取word文档
  9. SpringMVC-获得Restful风格的参数
  10. Springboot项目中配置tomcta监控日志