LOOK ! SpringBoot的外部化配置最全解析

如果没有profile指定的文件于profile指定的文件的配置属性同时定义,那么指定profile的配置优先。

5、使用占位符

===========================================================================

在使用application.properties中的值的时候,他们会从Environment中获取值,那就意味着,可以引用之前定义过的值,比如引用系统属性。具体做法如下:

name=天乔巴夏

description=${name} is my name

6、加密属性

==========================================================================

Spring Boot不提供对加密属性值的任何内置支持,但是,它提供了 修改Spring环境中的值所必需的挂钩点。我们可以通过实现EnvironmentPostProcessor接口在应用程序启动之前操纵Environment。

可以参考 howto.html ,查看具体使用方法。

7、使用YAML代替properties

========================================================================================

YAML是JSON的超集,是一种 指定层次结构配置数据的便捷格式 ,我们以properties文件对比一下就知道了:

#properties

spring.datasource.driver-class-name=com.mysql.cj.jdbc.Driver

spring.datasource.url=jdbc:mysql://localhost:3306/test?serverTimezone=GMT%2B8

spring.datasource.username=root

spring.datasource.password=123456

my.servers[0]=www.hyh.com

my.servers[1]=www.yhy.com

yml

spring:

datasource:

driver-class-name: com.mysql.cj.jdbc.Driver

url: jdbc:mysql://localhost:3306/test?serverTimezone=GMT%2B8

username: root

password: 123456

my:

server:

  • www.hyh.com

  • www.yhy.com

只要在类路径上具有SnakeYAML库,SpringApplication类就会自动支持YAML作为属性配置的方式。SpringBoot项目中的 spring-boot-starter 已经提供了相关类库: org.yaml.snakeyaml,因此SpringBoot天然支持这种方式配置。

关于yaml文件的格式,可以参考官方文档: Using YAML Instead of Properties

8、类型安全的属性配置

===============================================================================

上面说到通过 @Value(“${property}”) 注解来注入配置有时会比较麻烦,特别是当多个属性本质上具有层次结构的时候。SpringBoot提供了一种解决方案: 让强类型的bean管理和验证你的配置

直接来看具体的使用叭:

@ConfigurationPropertie定义一个绑定配置的JavaBean

============================================================================================================

  1. 使用默认构造器+getter和setter注入

@ConfigurationProperties(“acme”)

public class AcmeProperties {

private boolean enabled; //acme.enabled 默认为false

private InetAddress remoteAddress;// acme.remote-address 可以从String转换而来的类型

private final Security security = new Security();

//… 省略getter和setter方法

public static class Security {

private String username; // acme.security.username

private String password; // acme.security.password

private List roles = new ArrayList<>(Collections.singleton(“USER”));// acme.security.roles

//… 省略getter setter方法

}

}

这种方式依赖于默认的空构造函数,通过getter和setter方法赋值,因此getter和setter方法是必要的,且不支持静态属性的绑定。

如果嵌套pojo属性已经被初始化值: private final Security security = new Security(); 可以不需要setter方法。如果希望绑定器使用其默认构造函数动态创建实例,则需要setter。

  1. 通过@ContructorBinding注解使用构造器绑定的方式:

@ConstructorBinding //标注使用构造器绑定

@ConfigurationProperties(“acme”)

public class AcmeProperties {

private final Security security;

private final boolean enabled;

private final InetAddress remoteAddress;

public AcmeProperties(boolean enabled, InetAddress remoteAddress, Security security) {

this.enabled = enabled;

this.remoteAddress = remoteAddress;

this.security = security;

}

//…省略getter方法

@ToString

public static class Security {

private final String username;

private final String password;

private final List roles;

public Security(String username, String password,

@DefaultValue(“USER”) List roles) {

this.username = username;

this.password = password;

this.roles = roles;

}

}

//…省略getter方法

}

如果没有配置Security实例属性,那么最后结果:Security=null。如果我们想让Security={username=null,password=null,roles=[USER]},可以在Security上加上@DefaultValue。public AcmeProperties(boolean enabled, InetAddress remoteAddress, @DefaultValue Security security)

通过@EnableConfigurationProperties注册

======================================================================================================

已经定义好了JavaBean,并与配置属性绑定完成,接着需要注册这些bean。我们通常用的@Component或@Bean,@Import加载bean的方式在这里是不可取的,SpringBoot提供了解决方案: 使用@EnableConfigurationProperties ,我们既可以一一指定配置的类,也可以按照组件扫描的方式进行配置。

@SpringBootApplication

@EnableConfigurationProperties({HyhConfigurationProperties.class, MyProperties.class,AcmeProperties.class})

public class SpringBootProfileApplication {

}

@SpringBootApplication

@ConfigurationPropertiesScan({“com.hyh.config”})

public class SpringBootProfileApplication {

}

配置yaml文件

============================================================================

acme:

remote-address: 192.168.1.1

security:

username: admin

roles:

  • USER

  • ADMIN

注入properties,测试

===================================================================================

@Configuration

public class Application implements CommandLineRunner {

@Autowired

private AcmeProperties acmeProperties;

@Override

public void run(String… args) throws Exception {

System.out.println(acmeProperties);

}

}

//输出:

AcmeProperties(security=AcmeProperties.Security(username=admin, password=null, roles=[USER, ADMIN]), enabled=false, remoteAddress=/192.168.1.1)

宽松绑定

========================================================================

SpringBoot采用宽松的规则进行Environment和@ConfigurationProperties标注bean的匹配。如:

@ConfigurationProperties(prefix=“acme.my-project.person”)

public class OwnerProperties {

private String firstName;

public String getFirstName() {

return this.firstName;

}

public void setFirstName(String firstName) {

this.firstName = firstName;

}

}

下面表格中的属性名都可以匹配:

LOOK !  SpringBoot的外部化配置最全解析

@ConfigurationProperties注解中的prefix值必须是kebab case形式的,以 - 为分割符。

Spring官方建议,属性尽可能以lower-case kebab的形式:my.property-name=acme

Map如何绑定

===========================================================================

绑定到Map属性时,如果key包含 小写字母数字字符或-以外的任何其他字符 ,则需要使用方括号包围key,以便保留原始值。 如果键没有被 [] 包围,则所有非字母数字或-的字符都将被删除。如下:

hyh:

username: 天乔巴夏

password: 123456

map:

“[/key1]”: value1 #用引号包围[],用[]包围key

/key3: value3

key-4: value4

key/5: value5

结果:“map”:{/key1=value1,key5=value5, key-4=value4, key3=value3}

环境变量如何绑定

============================================================================

遵循三条原则:

  1. 把 . 换成下划线 _ 。

  2. 移除 - 。

  3. 小写转大写。

如: spring.main.log-startup-info 转为: SPRING_MAIN_LOGSTARTUPINFO , my.acme[0].other 转为 MY_ACME_0_OTHER 。

9、复杂类型

==========================================================================

之前介绍yml文件,介绍了单纯的数组形式或值的绑定,SpringBoot还支持复杂类型的绑定。

merge:

list:

  • name: 天乔巴夏

desc: 帅啊

  • name: tqbx

desc: 很帅啊

map:

key1:

name: summerday

desc: handsome!

key2:

name: summer

@ToString

@ConfigurationProperties(prefix = “merge”)

public class MergeProperties {

private final List list = new ArrayList<>();

private final Map<String,User> map = new HashMap<>();

public List getList() {

return list;

}

public Map<String, User> getMap() {

return map;

}

}

最后输出:

MergeProperties(

list=[User(name=天乔巴夏, desc=帅啊),

User(name=tqbx, desc=很帅啊)],

map={key1=User(name=summerday, desc=handsome!),

key2=User(name=summer, desc=null)}a

)

10、参数校验

===========================================================================

对@ConfigurationProperties类使用Spring的 @Valid 注解时,Spring Boot就会尝试对其进行验证。

你可以直接在配置类上使用JSR-303 javax.validation 约束注解。这个做法的前提是,你的类路径上有兼容的JSR-303实现:

org.hibernate

hibernate-validator

6.0.18.Final

然后将约束注解加到字段上,如下:

@Data

@Validated

@ConfigurationProperties(prefix = “validate”)

public class ValidateProperties {

@NotNull

private String name;

@Valid

private final SubProperties subProperties = new SubProperties();

@Data

public static class SubProperties {

@Min(value = 10,message = “年龄最小为10”)

public Integer age;

}

}

配置如下:

validate:

name: hyh

sub-properties:

age: 5

结果如下:

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值