【SpringBoot】配置文件的加载与属性值的绑定

简介: 【SpringBoot】配置文件的加载与属性值的绑定

Part1Question


在使用SpringBoot过程中你是否会有以下疑问?

  • 具体有多少种配置属性源的方式呢?
  • 为何使用@Value 注解就能够获取到属性源中的值呢?
  • 属性源这么多,如果属性相同的话 那么用哪个值呢?
  • 属性源是如何绑定到我们的程序中的呢?

本篇文章会针对以上问题逐个分析

Part2Answer


我们的所有属性源都存放在AbstractEnvironment中的属性propertySources中; 每加载一个属性源就会往里面塞一个propertySource; 然后当我们需要取某个属性的时候,就会从这个propertySources遍历查找,找到就返回; 所以我们就可以知道,如果多个属性源中有相同的属性,那么肯定是排在最前面的被找到就会返回,优先级最高; 那么这是整个背景; 我们现在来分析具体的问题

1具体有多少种配置属性源的方式呢?

以下优先级由高到低

  1. 命令行方式java -jar xx.jar --spring.profiles.active=pro&关于命令行的详细请看文章 【SpringBoot 一】SpringApplication启动类的Args详解
  2. 如果是以web方式启动的还会有 {servletConfigInitParamsservletContextInitParams}
  3. spring.application.json 外部json配置 在启动之初,SpringBoot会去当前的属性源(这个时候还只有systemPropertiessystemEnvironment)中查找有没有spring.application.json或者SPRING_APPLICATION_JSON的属性值;如果有则会把对应的值按照Json的格式解析成对应的属性源;例如:java -jar xx.jar --spring.application.json='{"foo":"bar"}'java -jar xx.jar -Dspring.application.json={\"foo\":\"888\"}如果这2种方式都用,那么以第一种命令行的方式为准,它的优先级更高
  4. systemProperties  JVM属性源; 使用方式就是java -jar xx.jar -Dmyname=src
  5. systemEnvironment系统环境变量属性源
  6. random随机数属性源 RandomValuePropertySource我们可以通过获取属性key =  random.int 来获取随机值
  7. 配置文件属性源 application.properties这样的配置文件
  8. 注解@PropertySources的属性源
  9. 通过SpringApplication.setDefaultProperties声明的默认属性源;

|方式  |用法  |描述| |--|--|--|--| | 命令行方式(启动的Args参数) | java -jar xx.jar --spring.profiles.active=pro |args用法详解| |spring.application.json 外部json配置|java -jar xx.jar --spring.application.json='{"foo":"bar"}'   java -jar xx.jar -Dspring.application.json={"foo":"888"}|在启动之初,SpringBoot会去当前的属性源(这个时候还只有systemProperties、systemEnvironment)中查找有没有spring.application.json或者SPRING_APPLICATION_JSON的属性值;如果有则会把对应的值按照Json的格式解析成对应的属性源| |JVM属性源|java -jar xx.jar -Dmyname=src| | |系统环境变量属性源|自动读取环境变量属性| |随机数属性源 RandomValuePropertySource|random.int 、random.long、random.int.5,100; 、|在SpringBoot中使用以上key可以获得指定的随机值| |配置文件application.properties||| |注解@PropertySources的属性源|可以把属性配置在另外单独的文件中,使用注解也可以加载为属性源|| |SpringApplication.setDefaultProperties声明的默认属性源|||

2属性源这么多,如果属性相同的话 那么用哪个值呢?

属性源是一个List,读取的时候是遍历List; 先读取到的立马返回; 优先级的顺序是上面1-9种方式;

3为何使用@Value 注解就能够获取到属性源中的值呢?

我们先介绍一下@Value的几种常用用法

       //常量
   @Value("#{1}")
   privateint constant;

   //从属性源取值
   @Value("${test.name}")
   private String name;

   //从属性源取值
   @Value("${test.name2: defaultname}")
   private String namedefault;

   //从容器中获取bean的的属性值
   @Value("#{developerProperty.name}")
   private String dname;

   //从指定属性源获取属性值(jvm属性)
   @Value("#{systemProperties['spring.application.json']}")
   private String systemPropertiesjson;

   //从指定属性源获取属性值(系统环境属性源)
   @Value("#{systemEnvironment['HOME']}")
   private String systemEnvironmentHOME;

   //从指定属性源获取属性值 默认值
   @Value("#{systemEnvironment['HOME22']?:'default'}")
   private String systemEnvironmentHOMEdefault;

   //获取随机值
   @Value("${random.int.5,100;}")
   private Integer randomint;

SpringBoot 中 @Value 源码解析

4属性源是如何绑定到我们的程序中的呢?

先看看用法;  下面是SpringBoot启动过程中 将配置spring.main开头的属性 绑定到 SpringApplication中的用法

protectedvoidbindToSpringApplication(ConfigurableEnvironment environment) {
 try {
  Binder.get(environment).bind("spring.main", Bindable.ofInstance(this));
 }
 catch (Exception ex) {
  thrownew IllegalStateException("Cannot bind to SpringApplication", ex);
 }
}

绑定到实例中

那我们自己来写一个demo将配置文件的属性值绑定到某个类实例中;

publicclassBinderTest {


   private String bname;

   private Integer bage;

   private BinderInnerTest binderInnerTest;


   public String getBname() {
       return bname;
   }

   publicvoidsetBname(String bname) {
       this.bname = bname;
   }

   public Integer getBage() {
       return bage;
   }

   publicvoidsetBage(Integer bage) {
       this.bage = bage;
   }

   public BinderInnerTest getBinderInnerTest() {
       return binderInnerTest;
   }

   publicvoidsetBinderInnerTest(BinderInnerTest binderInnerTest) {
       this.binderInnerTest = binderInnerTest;
   }

   publicstaticclassBinderInnerTest{

       private String innerName;

       private Integer innerage;

       public String getInnerName() {
           return innerName;
       }

       publicvoidsetInnerName(String innerName) {
           this.innerName = innerName;
       }

       public Integer getInnerage() {
           return innerage;
       }

       publicvoidsetInnerage(Integer innerage) {
           this.innerage = innerage;
       }
   }


}


PS:上面要注意. set get 不能少,  而且如果是内部类,必须要是  public static class 否则内部类的属性不会正确绑定的!

配置文件

binder.test.bname=bindname

binder.test.bage=18

binder.test.binderInnerTest.innerName=bindInnername

binder.test.binderInnerTest.innerage=28

绑定


BindResult<BinderTest> result = Binder.get(environment).bind("binder.test", Bindable.of(BinderTest.class));
System.out.println(result);

绑定成功为何 binder.test 这种前缀就能把实例属性给绑定上呢? Binder属性绑定源码解析 TODO。。。。

有没有觉得这种方式很熟悉?SpringBoot 中有个注解@ConfigurationProperties(prefix = "") 的功能是不差不多?也是将属性值绑定到实例中去; 那么它是怎么实现的呢? 是不是也是通过Binder的方式实现的? 答案是肯定的,贴一个关键代码 ConfigurationPropertiesBinder

publicvoidbind(Bindable<?> target) {
 ConfigurationProperties annotation = target
   .getAnnotation(ConfigurationProperties.class);
 Assert.state(annotation != null,
   () -> "Missing @ConfigurationProperties on " + target);
 List<Validator> validators = getValidators(target);
 BindHandler bindHandler = getBindHandler(annotation, validators);
 getBinder().bind(annotation.prefix(), target, bindHandler);
}

详细分析 ConfigurationProperties TODO..

绑定到List中

配置文件

binder.test2.list[0]=valuea

binder.test2.list[1]=valueb

binder.test2.list[2]=valuec

binder.test2.list[3]=valued

PS: 数组index 必须连续

绑定

       
   BindResult<List<String>> resultlist = Binder.get(environment).bind("binder.test2.list", Bindable.listOf(String.class));

绑定成功

绑定到Map中

配置文件

binder.test3.a=a
binder.test3.b=b
binder.test3.c=c

绑定


   BindResult<Map<String, String>> resultmap = Binder.get(environment).bind("binder.test3", Bindable.mapOf(String.class,String.class));

绑定成功

PS: 如果多个属性源中有相同的属性源前缀会如何?那么会按照属性源的优先级绑定;后面的不再绑定

相关文章
|
6月前
|
存储 Java 文件存储
微服务——SpringBoot使用归纳——Spring Boot使用slf4j进行日志记录—— logback.xml 配置文件解析
本文解析了 `logback.xml` 配置文件的详细内容,包括日志输出格式、存储路径、控制台输出及日志级别等关键配置。通过定义 `LOG_PATTERN` 和 `FILE_PATH`,设置日志格式与存储路径;利用 `&lt;appender&gt;` 节点配置控制台和文件输出,支持日志滚动策略(如文件大小限制和保存时长);最后通过 `&lt;logger&gt;` 和 `&lt;root&gt;` 定义日志级别与输出方式。此配置适用于精细化管理日志输出,满足不同场景需求。
1322 1
|
6月前
|
Java 测试技术 微服务
微服务——SpringBoot使用归纳——Spring Boot中的项目属性配置——少量配置信息的情形
本课主要讲解Spring Boot项目中的属性配置方法。在实际开发中,测试与生产环境的配置往往不同,因此不应将配置信息硬编码在代码中,而应使用配置文件管理,如`application.yml`。例如,在微服务架构下,可通过配置文件设置调用其他服务的地址(如订单服务端口8002),并利用`@Value`注解在代码中读取这些配置值。这种方式使项目更灵活,便于后续修改和维护。
85 0
|
6月前
|
Java 数据库 微服务
微服务——SpringBoot使用归纳——Spring Boot中的项目属性配置——指定项目配置文件
在实际项目中,开发环境和生产环境的配置往往不同。为简化配置切换,可通过创建 `application-dev.yml` 和 `application-pro.yml` 分别管理开发与生产环境配置,如设置不同端口(8001/8002)。在 `application.yml` 中使用 `spring.profiles.active` 指定加载的配置文件,实现环境快速切换。本节还介绍了通过配置类读取参数的方法,适用于微服务场景,提升代码可维护性。课程源码可从 [Gitee](https://gitee.com/eson15/springboot_study) 下载。
194 0
|
6月前
|
Java 微服务 Spring
微服务——SpringBoot使用归纳——Spring Boot中的项目属性配置——少量配置信息的情形
在微服务架构中,随着业务复杂度增加,项目可能需要调用多个微服务。为避免使用`@Value`注解逐一引入配置的繁琐,可通过定义配置类(如`MicroServiceUrl`)并结合`@ConfigurationProperties`注解实现批量管理。此方法需在配置文件中设置微服务地址(如订单、用户、购物车服务),并通过`@Component`将配置类纳入Spring容器。最后,在Controller中通过`@Resource`注入配置类即可便捷使用,提升代码可维护性。
86 0
|
10月前
|
监控 IDE Java
如何在无需重新启动服务器的情况下在 Spring Boot 上重新加载我的更改?
如何在无需重新启动服务器的情况下在 Spring Boot 上重新加载我的更改?
841 8
|
10月前
|
Java 数据库连接 数据库
springboot启动配置文件-bootstrap.yml常用基本配置
以上是一些常用的基本配置项,在实际应用中可能会根据需求有所变化。通过合理配置 `bootstrap.yml`文件,可以确保应用程序在启动阶段加载正确的配置,并顺利启动运行。
1170 2
|
10月前
|
Java Spring 容器
SpringBoot读取配置文件的6种方式,包括:通过Environment、@PropertySource、@ConfigurationProperties、@Value读取配置信息
SpringBoot读取配置文件的6种方式,包括:通过Environment、@PropertySource、@ConfigurationProperties、@Value读取配置信息
1860 3
|
11月前
|
druid Java Maven
|
11月前
|
架构师 Java 开发者
得物面试:Springboot自动装配机制是什么?如何控制一个bean 是否加载,使用什么注解?
在40岁老架构师尼恩的读者交流群中,近期多位读者成功获得了知名互联网企业的面试机会,如得物、阿里、滴滴等。然而,面对“Spring Boot自动装配机制”等核心面试题,部分读者因准备不足而未能顺利通过。为此,尼恩团队将系统化梳理和总结这一主题,帮助大家全面提升技术水平,让面试官“爱到不能自已”。
得物面试:Springboot自动装配机制是什么?如何控制一个bean 是否加载,使用什么注解?
|
XML 设计模式 Java
springboot创建并配置环境3 - 配置扩展属性(下)
springboot创建并配置环境3 - 配置扩展属性(下)
springboot创建并配置环境3 - 配置扩展属性(下)