SpringBoot启动如何加载application.yml配置文件

一、前言
在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自动加载配置文件的机制!

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

springboot什么时候加载配置文件
springboot通过哪个类加载配置文件
springboot自动加载配置文件流程
激活文件优先级
文件加载路径优先级
文件后缀优先级
二、提出猜想
我们知道在使用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

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

active与默认优先级
文件路径优先级
文件后缀优先级
未完待续。。。
四、提问
springboot学习遗留问题,
1.active和默认的谁覆盖谁
2.flter区别
3.多个配置文件如何覆盖

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/250106.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

阿里云服务器端口开放对外访问权限

登陆阿里云管理控制台 点击自己的实例 点击安全组配置 点击配置规则 点击添加安全组规则 配置出入放心&#xff0c;和开放的端口号&#xff0c;以及那些网段可以访问&#xff0c;这里设置所有网段都可以访问 转自&#xff1a;https://jingyan.baidu.com/article/95c9d20d624d1e…

PageHelper工作原理

数据分页功能是我们软件系统中必备的功能&#xff0c;在持久层使用mybatis的情况下&#xff0c;pageHelper来实现后台分页则是我们常用的一个选择&#xff0c;所以本文专门类介绍下。 PageHelper原理 相关依赖 <dependency><groupId>org.mybatis</groupId>&…

10-多写一个@Autowired导致程序崩了

再是javaweb实验六中&#xff0c;是让我们改代码&#xff0c;让它跑起来&#xff0c;结果我少注释了一个&#xff0c;导致一直报错&#xff0c;检查许久没有找到&#xff0c;最后通过代码替换逐步查找&#xff0c;才发现问题。 转载于:https://www.cnblogs.com/zhumengdexiaoba…

springboot---整合redis

pom.xml新增 <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-data-redis</artifactId></dependency>代码结构如下 其中redis.yml是连接redis的配置文件&#xff0c;RedisConfig.java是java配置…

[Head First Java] - Swing做一个简单的客户端

参考 - P487 1. vscode配置java的格式 点击左下角齿轮 -> 设置 -> 打开任意的setting.json输入如下代码 {code-runner.executorMap": {"java": "cd $dir && javac -encoding utf-8 $fileName && java $fileNameWithoutExt"},…

计算机网络知识总结

一 OSI与TCP/IP各层的结构与功能&#xff0c;都有哪些协议 OSI的七层体系结构概念清楚&#xff0c;理论也很完整&#xff0c;但是它比较复杂而且不实用。在这里顺带提一下之前一直被一些大公司甚至一些国家政府支持的OSI失败的原因&#xff1a; OSI的专家缺乏实际经验&#xff…

[Head First Java] - 给线程命名

参考 - P503 public class RunThreads implements Runnable {public static void main (String[] args) {RunThreads runner new RunThreads();Thread alpha new Thread(runner);Thread beta new Thread(runner);alpha.setName("Alpha thread");beta.setName(&qu…

快速排序的C++版

int Partition(int a[], int low, int high) {int x a[high];//将输入数组的最后一个数作为主元&#xff0c;用它来对数组进行划分int i low - 1;//i是最后一个小于主元的数的下标for (int j low; j < high; j)//遍历下标由low到high-1的数{if (a[j] < x)//如果数小于…

asp.net中提交表单数据时提示从客户端(。。。)中检测到有潜在危险的 Request.Form 值...

看到这个图是不是很亲切熟悉哈&#xff0c;做过。net的肯定都见过哈 已经 将近4年没碰。net了&#xff0c;今天正好朋友的程序有几个bug,让我帮忙修复下&#xff0c;于是我就抱着试试看的心情改了改&#xff0c;改到最后一个问题的时候也就是上面的这个问题&#xff0c;我一看&…

Shiro表结构设计

表设计 开发用户-角色-权限管理系统&#xff0c;首先我们需要知道用户-角色-权限管理系统的表结构设计。 在用户-角色-权限管理系统找那个一般会涉及5张表&#xff0c;分别为&#xff1a; 1.sys_users用户表 2.sys_roles角色表 3.sys_permissions权限表&#xff08;或资源表&…

[Java核心技术(卷I)] - 简易的日历

参考 - P102~P103 1. 目标 生成一个日历,格式如下图所示。 ps: 当前的天数需要标记为* 2. 核心 对日历的变量 import java.time.*; public class CalendarTest{public static void main(String[] args) {LocalDate date LocalDate.now(); // 获取当前日期int month date…

个人作业——福大微信公众号使用评测

案例分析&#xff1a;在福州大学公众号上&#xff0c;我们可以即时使用手机关注福大新闻&#xff0c;查看自身课表、成绩等。公众号可能存在一些小bug影响同学们的用户体验。本次作业中&#xff0c;作为一个用户——福大的学生&#xff0c;将切身体验该公众号的功能&#xff0c…

在winform中使用wpf窗体

在winform项目&#xff0c;通过引用dll可以添加WPF窗体&#xff0c;如下 但是如果直接在winform的项目中添加wpf窗体还是有部分问题&#xff0c;图片的显示。 直接在XAML界面中用Source属性设置图片会出现错误。必须通过后台代码的方式来实现。 image1.Source GetImageIcon(gl…

shiro---注解

RequiresAuthentication 验证用户是否登录&#xff0c;等同于方法subject.isAuthenticated() 结果为true时。 RequiresUser 验证用户是否被记忆&#xff0c;user有两种含义&#xff1a; 一种是成功登录的&#xff08;subject.isAuthenticated() 结果为true&#xff09;&…

[Java核心技术(卷I)] - Java中的参数能做什么和不能做什么

1. 参考 - P123 ~ P126 2. 你将学到 Java中对方法参数能做什么和不能做什么 方法不能修改基本数据类型的参数(数值型或布尔型)方法可以改变对象参数的状态方法不能让一个对象参数引用一个新的对象 3. 代码证明 public class ParamTest {public static void main(String[] ar…

软件构造 第五章第一节 可复用性的度量、形态和外部观察

第五章第一节 可复用性的度量、形态和外部观察 面向复用编程(programming for reuse)&#xff1a;开发出可复用的软件 基于复用编程(programming with reuse)&#xff1a;利用已有的可复用软件搭建应用系统 代码复用的类型&#xff1a; 白盒复用&#xff1a;源代码可见&#x…

[web性能优化] - 使用在线工具对html、js、css进行压缩

参考 1. 学习点 使用 在线工具对html、css、js进行压缩学会分析压缩前后的效率提高点 2. 解决方案: 2.1 HTML压缩 在线压缩nodejs提供了 html-minifier工具(在构建层对代码进行压缩)后端模板引擎渲染压缩 2.2 CSS压缩 使用html-minifier对html中的css进行压缩使用clean-cs…

SpringBoot之基础

简介 背景 J2EE笨重的开发 / 繁多的配置 / 低下的开发效率 / 复杂的部署流程 / 第三方技术集成难度大 特点 ① 快速创建独立运行的spring项目以及主流框架集成 ② 使用嵌入式的Servlet容器, 应用无需达成war包 ③ starters自动依赖和版本控制 ④ 大量自动配置, 简化开发, 也可修…

[Java核心技术(卷I)] - vscode手动编译运行继承类

参考 - P160~P161 主要有3个类: 一个测试类(ManagerTest)、一个子类(Manager)、一个父类(Employee) 注意点: -1. 使用 javac -d . *.java进行预编译 目录结构入下: 此时会生成目录结构如下: 之后运行 java com.inheritance.ManagerTest 附上几个类的代码 // com.inhe…

【Social Listening实战】当数据分析遭遇心理动力学:用户深层次的情感需求浮出水面...

本文转自知乎 作者&#xff1a;苏格兰折耳喵 ————————————————————————————————————————————————————— 本文篇幅较长&#xff0c;分为五部分&#xff0c;在中间部分有关于心理分析工具的介绍&#xff0c;案例分散在第二部…