简介
本文将会详细讲解 @ConfigurationProperties 在 Spring Boot 中的使用。
添加依赖关系
首先我们需要添加 Spring Boot 依赖:
<parent>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-parent</artifactId>
<relativePath/> <!-- lookup parent from repository -->
</parent>
一个简单的例子
@ConfigurationProperties 需要和 @Configuration 配合使用,我们通常在一个 POJO 里面进行配置:
@Data
@Configuration
@ConfigurationProperties(prefix = "mail")
public class ConfigProperties {
private String hostName;
private int port;
private String from;
}
上面的例子将会读取 properties 文件中所有以 mail 开头的属性,并和 bean 中的字段进行匹配:
#Simple properties
[email protected]
mail.port=9000
[email protected]
Spring 的属性名字匹配支持很多格式,如下所示所有的格式都可以和 hostName 进行匹配:
mail.hostName
mail.hostname
mail.host_name
mail.host-name
mail.HOST_NAME
如果你不想使用 @Configuration, 那么需要在 @EnableConfigurationProperties 注解中手动导入配置文件如下:
@SpringBootApplication
@EnableConfigurationProperties(ConfigProperties.class)
public class ConfigPropApp {
public static void main(String[] args) {
SpringApplication.run(ConfigPropApp.class,args);
}
}
我们也可以在 @ConfigurationPropertiesScan 中指定 Config 文件的路径:
@SpringBootApplication
@ConfigurationPropertiesScan("com.flydean.config")
public class ConfigPropApp {
public static void main(String[] args) {
SpringApplication.run(ConfigPropApp.class,args);
}
}
这样的话程序只会在 com.flydean.config 包中查找 config 文件。
属性嵌套
我们可以嵌套 class,list,map, 下面我们举个例子,先创建一个普通的 POJO:
@Data
public class Credentials {
private String authMethod;
private String username;
private String password;
}
然后创建一个嵌套的配置文件:
@Data
@Configuration
@ConfigurationProperties(prefix = "nestmail")
public class NestConfigProperties {
private String host;
private int port;
private String from;
private List<String> defaultRecipients;
private Map<String, String> additionalHeaders;
private Credentials credentials;
}
对应的属性文件如下:
# nest Simple properties
[email protected]
nestmail.port=9000
[email protected]
#List properties
nestmail.defaultRecipients[0][email protected]
nestmail.defaultRecipients[1][email protected]
#Map Properties
nestmail.additionalHeaders.redelivery=true
nestmail.additionalHeaders.secure=true
#Object properties
nestmail.credentials.username=john
nestmail.credentials.password=password
nestmail.credentials.authMethod=SHA1
@ConfigurationProperties 和 @Bean
@ConfigurationProperties 也可以和 @Bean 一起使用如下所示:
@Data
public class Item {
private String name;
private int size;
}
看下怎么使用:
@Data
@Configuration
public class BeanConfigProperties {
@Bean
@ConfigurationProperties(prefix = "item")
public Item item() {
return new Item();
}
}
属性验证
@ConfigurationProperties 可以使用标准的 JSR-303 格式来做属性验证。我们举个例子:
如果我们的属性不满足上诉条件,可能出现如下异常:
属性转换
@ConfigurationProperties 也支持多种属性转换,下面我们以 Duration 和 DataSize 为例:
我们定义两个 Duration 的字段:
@ConfigurationProperties(prefix = "conversion")
public class PropertyConversion {
private Duration timeInDefaultUnit;
private Duration timeInNano;
...
}
在属性文件中定义这两个字段:
conversion.timeInDefaultUnit=10
conversion.timeInNano=9ns
我们看到上面的属性可以带单位的。可选的单位是:ns, us, ms, s, m, h 和 d,分别对应纳秒,微妙,毫秒,秒,分钟,小时和天。默认单位是毫秒。我们也可以在注解中指定单位:
@DurationUnit(ChronoUnit.DAYS)
private Duration timeInDays;
对应的配置文件如下:
conversion.timeInDays=2
下面我们再看看 DataSize 怎么使用:
private DataSize sizeInDefaultUnit;
private DataSize sizeInGB;
@DataSizeUnit(DataUnit.TERABYTES)
private DataSize sizeInTB;
对应的属性文件:
conversion.sizeInDefaultUnit=300
conversion.sizeInGB=2GB
conversion.sizeInTB=4
Datasize 支持 B, KB, MB, GB 和 TB。
自定义 Converter
同样的 Spring Boot 也支持自定义属性转换器。我们先定义一个 POJO 类:
public class Employee {
private String name;
private double salary;
}
对应的属性文件:
conversion.employee=john,2000
我们需要自己实现一个 Converter 接口的转换类:
@Component
@ConfigurationPropertiesBinding
public class EmployeeConverter implements Converter<String, Employee> {
@Override
public Employee convert(String from) {
String[] data = from.split(",");
return new Employee(data[0], Double.parseDouble(data[1]));
}
}
本文的例子可以参看: https://github.com/ddean2009/learn-springboot2/tree/master/springboot-ConfigurationProperties
更多教程请参考 flydean 的博客