服务器之家:专注于VPS、云服务器配置技术及软件下载分享
分类导航

PHP教程|ASP.NET教程|Java教程|ASP教程|编程技术|正则表达式|C/C++|IOS|C#|Swift|Android|VB|R语言|JavaScript|易语言|vb.net|

服务器之家 - 编程语言 - Java教程 - 美团二面:SpringBoot读取配置优先级顺序是什么?

美团二面:SpringBoot读取配置优先级顺序是什么?

2024-04-15 15:47码农Academy Java教程

启动Spring Boot应用时,可以直接通过命令行参数来覆盖或设置配置属性。命令行参数通常以--​开头,后面紧跟属性名和值,如--server.port=8080。这种方式可以在不修改配置文件的前提下临时调整应用配置。

引言

Spring Boot作为一种轻量级的Java应用程序框架,以其开箱即用、快速搭建新项目的特性赢得了广大开发者的青睐。其核心理念之一就是简化配置过程,使开发者能够快速响应复杂多变的生产环境需求。为了实现这一点,Spring Boot支持丰富的外部化配置机制,允许应用程序根据不同的部署环境灵活加载相应的配置属性,而无需修改代码本身。

在Spring Boot生态系统中,配置属性可以从各种来源获取,比如:Java属性文件、YAML文件、环境变量、命令行参数等。这些配置属性能够在运行时动态注入到Bean中,极大地提高了系统的可扩展性和可配置性。然而,为了确保一致性和防止配置冲突,Spring Boot在加载这些外部配置时遵循一套严格的优先级顺序。掌握这套优先级规则至关重要,因为它直接影响着最终生效的配置属性值,进而决定了应用程序的行为模式。

本文将深入探讨Spring Boot加载外部配置属性的优先级规则,详尽梳理各个配置源的加载顺序,并结合实际应用场景举例说明,以便我们能够更高效地管理和迁移配置,确保在不同环境下应用程序都能稳定、准确地运行。

Spring Boot外部化配置概述

Spring Boot的核心价值之一在于其强大的外部化配置能力,这使得应用程序能够在不改变代码的情况下适应不同的运行环境。外部化配置意味着将应用程序的关键配置信息移至应用程序代码之外,便于根据不同环境(如开发、测试、生产等)进行定制化配置。Spring Boot提供了多样化的外部配置源以及便捷的属性注入方式,使得这种配置机制变得异常灵活且易于管理。

多样化配置源

Spring Boot支持多种类型的外部配置源,主要有如下几个方面:

Properties文件: 通常使用.properties格式,采用键值对的形式存储配置信息。

server.port=8080
logging.level.root=DEBUG

YAML文件: 相较于传统的properties文件,YAML提供了更直观、层次更分明的数据结构,尤其适合存储复杂配置。使用.yml格式。

server:
  port: 8080
logging:
  level:
    root: DEBUG

1. 环境变量: 操作系统级别的环境变量可以被Spring Boot识别并作为配置源,这对于云环境和容器化部署尤为实用。

2. 命令行参数: 启动Spring Boot应用时,可以传入命令行参数(以--开头)直接覆盖已有配置。

属性注入方式

在Spring Boot中,外部配置的属性值可以通过以下几种方式方便地注入到Bean中。

• @Value注解:可以直接在字段或方法参数上使用此注解,将配置属性值注入到目标对象中。

• Environment接口:Spring框架提供的环境抽象类,可以用来查询所有已加载的配置信息。

• @ConfigurationProperties注解:用于绑定一组相关配置到一个专门的Java Bean中,提供更结构化的配置管理方式。

配置加载优先级

Spring Boot对来自不同配置源的同名属性可以按照一定的优先级顺序进行覆盖。其优先级从上到下变高,即后面的配置源将覆盖前面的配置源。

1. 默认属性(通过SpringApplication.setDefaultProperties方法设置)

2. @PropertySource注解加载的配置

3. Config Data(配置数据)(本地文件系统或打包在jar中的application.properties和application-{profile}.properties)

4. 特殊属性源(如随机数生成器、环境变量、系统属性、JNDI属性等)

5. Servlet容器相关的初始化参数

6. SPRING_APPLICATION_JSON格式的环境变量或系统属性

7. 命令行参数

8. 测试相关的属性注入方式(如@SpringBootTest、@DynamicPropertySource和@TestPropertySource)

以上优先级顺序来源于官网:Spring Boot Reference Documentation

Spring Boot配置加载顺序详解

默认属性

默认属性是指Spring Boot框架内置的一些默认配置值。可以在创建SpringApplication实例时,通过调用setDefaultProperties(Map<String, Object> defaultProperties)方法来提供一组默认属性,这些属性将被优先加载,但是也会被其他配置覆盖。

@SpringBootApplication
public class SpringBootBaseApplication {

    public static void main(String[] args) {
        Map<String, Object> defaultProperties = new HashMap<>();
        defaultProperties.put("server.port", "9000"); // 自定义默认端口
        SpringApplication app = new SpringApplication(SpringBootBaseApplication.class);
        app.setDefaultProperties(defaultProperties);
        app.run(args);
    }
}

美团二面:SpringBoot读取配置优先级顺序是什么?图片

@PropertySource注解

@PropertySource注解用于在Spring Boot的@Configuration类上加载外部属性文件。当我们在配置类上使用@PropertySource时,需要注意的是,这些属性源并不会立即被添加到Spring的Environment中。它们是在Spring应用上下文刷新(refresh)阶段才会被真正加载并合并到环境变量中。

有兴趣的可以跟一下源码,org.springframework.context.support.AbstractApplicationContext#invokeBeanFactoryPostProcessors中执行的。

Spring Boot的主引导配置,如服务器端口(server.port)、日志框架的初始化(例如日志级别设置)等,也是在应用上下文刷新之前就被读取并应用的。因此,对于这类早期就需要读取的配置,应该直接在application.properties或者环境变量等更早被加载的配置源中进行设置。

我们创建一个propertysource.properties文件:

server.port = 9001
coderacademy.name = CoderAcademy

然后我们在@Configuration配置上使用@PropertySource导入propertysource.properties文件。

@PropertySource(value = "classpath:propertysource.properties")
@Configuration
public class MyConfig {

}

我们在应用启动后看一下上述配置:

@SpringBootApplication
public class SpringBootBaseApplication {

    public static void main(String[] args) {
        Map<String, Object> defaultProperties = new HashMap<>();
        defaultProperties.put("server.port", "9000"); // 自定义默认端口
        SpringApplication app = new SpringApplication(SpringBootBaseApplication.class);
        app.setDefaultProperties(defaultProperties);
        ConfigurableApplicationContext context = app.run(args);
        Environment environment = context.getEnvironment();
        System.out.println("coderacademy.name: " + environment.getProperty("coderacademy.name"));
    }
}

打印结果:

可以看出server.port变成了9001,即@PropertySource加载的配置覆盖了SpringBoot默认的属性值。

Config Data(配置数据)

Config Data(配置数据)是Spring Boot中用于外部化应用配置的核心部分。主要由内部配置文件以及外部配置文件。

内部配置文件

内部配置文件最基础的应用配置文件,位于项目构建后的jar包内部。位于src/main/resource目录下的文件。

外部配置文件

可以将配置文件放在jar包外面的某个路径下。这种方式有助于在不修改jar包的情况下变更配置。比如我们使用的配置中心(nacos,apollo等),也可以通过spring.config.location或者spring.config.additional-location指定的文件等。

SpringBoot在启动时会默认从特定的目录中加载这些配置文件。我们可以从ConfigDataEnvironment中找到这些目录:

其目录的加载顺序由低到高为:

file:./
file:./config/
file:./config/*/
classpath:/
classpath:/config/

其中file代表应用根目录下的文件,而classpath为resources下的文件。

这些配置文件的配置优先级顺序由低到高为:

classpath:/
classpath:/config/
file:./
file:./config/
file:./config/*/

本例基于SpringBoot2.7版本。 关于SpringBoot加载内部配置文件的执行流程以及原理,请参考: 华为二面:SpringBoot读取_配置文件_的原理是什么?加载顺序是什么?

我们分别在这些目录下创建配置文件application.properties:

我们在对应文件中写入他们的目录路径:

1: config.data.path = classpath:./
2: config.data.path = classpath:./config/
3: config.data.path = file:./
4: config.data.path = file:./config/
5: config.data.path = file:./config/dev

我们在SpringBoot启动时打印config.data.path的值:

@SpringBootApplication
public class SpringBootConfigApplication {

    public static void main(String[] args) {
        Map<String, Object> defaultProperties = new HashMap<>();
        defaultProperties.put("server.port", "9000"); // 自定义默认端口
        SpringApplication app = new SpringApplication(SpringBootConfigApplication.class);
        app.setDefaultProperties(defaultProperties);
        ConfigurableApplicationContext context = app.run(args);
        Environment environment = context.getEnvironment();
        System.out.println("config.data.path: " + environment.getProperty("config.data.path"));
    }
}

我们分步进行验证,先验证1,2,打印结果:

config.data.path: classpath:./config/

继续验证1,2,3,打印结果:

config.data.path: file:./

验证1,2,3,4,打印结果:

config.data.path: file:./config/

验证1,2,3,4,5,打印结果:

config.data.path: file:./config/dev

随机值属性源

RandomValuePropertySource 在Spring Boot中,RandomValuePropertySource是一个特殊属性源,它并不来源于固定的配置文件或环境变量,而是由Spring Boot框架在启动时自动添加。这个属性源提供的属性名以random.*开头,可以用于生成随机值。例如,你可以在配置文件中引用random.int或random.long等属性,Spring Boot在启动时会为这些属性生成随机整数值。这对于需要在运行时生成一些临时或随机值的场景非常有用,如临时密码、缓存密钥等。

比如我们在application.properties中设置random.int=100

random.int=100

我们在SpringBoot启动时获取``random.int`的值:

@SpringBootApplication
public class ConfigApplication
{
    public static void main( String[] args )
    {
        SpringApplication app = new SpringApplication(ConfigApplication.class);
        ConfigurableApplicationContext context = app.run(args);
        Environment environment = context.getEnvironment();
        System.out.println("random.int: " + environment.getProperty("random.int"));
    }
}

打印结果为:

random.int: -510589238

并且每次重新启动应用,打印的结果都不一样。

操作系统环境变量

在Spring Boot中,环境变量可以用作配置源,Spring Boot会自动检测并加载这些环境变量作为应用的配置属性。例如,如果在操作系统中设置了环境变量MY_APP_PORT=8080,那么在Spring Boot应用中可以通过${MY_APP_PORT}来引用这个值。

我们设置环境变量为config.data.path=环境变量:

我们启动引用,依然打印config.data.path的结果为:

config.data.path: 环境变量

Java系统属性

Java系统属性是通过System.setProperty()方法设置一系列键值对。

@SpringBootApplication
public class ConfigApplication
{
    static {
        System.setProperty("config.data.path", "SystemProperty"); // 设置系统属性
    }

    public static void main( String[] args )
    {
        SpringApplication app = new SpringApplication(ConfigApplication.class);
        ConfigurableApplicationContext context = app.run(args);
        Environment environment = context.getEnvironment();
        System.out.println("config.data.path: " + environment.getProperty("config.data.path"));
    }
}

打印结果为:

config.data.path: SystemProperty

SPRING_APPLICATION_JSON环境变量中的内嵌JSON属性

SPRING_APPLICATION_JSON 是 Spring Boot 提供的一种机制,允许通过环境变量传递 JSON 格式的配置给应用程序。这个环境变量的内容会被解析成一个 JSON 对象,并合并到Spring的Environment中,就像其他属性源一样。

@SpringBootApplication
public class ConfigApplication
{
    static {
        System.setProperty("config.data.path", "SystemProperty"); // 设置系统属性
        System.setProperty("SPRING_APPLICATION_JSON", "{\"config.data.path\":\"SPRING_APPLICATION_JSON环境变量中的内嵌JSON属性\"}");
    }

    public static void main( String[] args )
    {
        SpringApplication app = new SpringApplication(ConfigApplication.class);
        ConfigurableApplicationContext context = app.run(args);
        Environment environment = context.getEnvironment();
        System.out.println("config.data.path: " + environment.getProperty("config.data.path"));
    }
}

打印结果:

config.data.path: SPRING_APPLICATION_JSON环境变量中的内嵌JSON属性

命令行参数

启动Spring Boot应用时,可以直接通过命令行参数来覆盖或设置配置属性。命令行参数通常以--开头,后面紧跟属性名和值,如--server.port=8080。这种方式可以在不修改配置文件的前提下临时调整应用配置。命令行参数具有较高的优先级,可以覆盖其它配置源中的属性值。

我们使用java -jar启动SpringBoot:

java -jar ./springboot-config-1.0-SNAPSHOT.jar --config.data.path=命令行参数

打印结果为:

config.data.path: 命令行参数

原文地址:https://mp.weixin.qq.com/s/V2m1j6f7U6RkguxiSWreWw

延伸 · 阅读

精彩推荐
  • Java教程SpringData Repository接口用法解析

    SpringData Repository接口用法解析

    这篇文章主要介绍了SpringData Repository接口用法解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参...

    IT-執念2542020-08-27
  • Java教程Java Socket+mysql实现简易文件上传器的代码

    Java Socket+mysql实现简易文件上传器的代码

    最近在做一个小项目,项目主要需求是实现一个文件上传器,通过客户端的登陆,把本地文件上传到服务器的数据库(本地的)。下面通过本文给大家分享...

    PettyKoKo4612020-06-27
  • Java教程Springboot网站第三方登录 微信登录

    Springboot网站第三方登录 微信登录

    这篇文章主要为大家详细介绍了Springboot网站第三方登录 ,微信登录,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一...

    摩天轮丶7612021-03-03
  • Java教程springboot如何获取文件流

    springboot如何获取文件流

    这篇文章主要介绍了springboot如何获取文件流,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教...

    空城v14192022-10-08
  • Java教程关于JavaEE内部类的部分注意事项

    关于JavaEE内部类的部分注意事项

    这篇文章主要介绍了关于JavaEE内部类的部分注意事项,将一个类定义在另一个类里面或者一个方法里面,这样的类称为内部类,这是一种封装思想,那么使用内部...

    Killing Vibe4242023-03-27
  • Java教程通过Java实现设置Word文档页边距的方法详解

    通过Java实现设置Word文档页边距的方法详解

    页边距是指页面的边线到文字的距离。通常可在页边距内部的可打印区域中插入文字和图形等。今天这篇文章将为您展示如何通过编程方式,设置Word 文档...

    Carina-baby10502023-03-04
  • Java教程jqGrid 学习笔记整理——进阶篇(一 )

    jqGrid 学习笔记整理——进阶篇(一 )

    这篇文章主要介绍了jqGrid 学习笔记整理——进阶篇(一 )的相关资料,需要的朋友可以参考下 ...

    XeonMic5712020-04-17
  • Java教程postman测试传入List<String>参数方式

    postman测试传入List<String>参数方式

    这篇文章主要介绍了postman测试传入List<String>参数方式,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教...

    烟萝5822021-11-23