Spring Boot 配置文件
Spring Boot 官方提供了两种常用的配置文件格式,分别是properties、yml格式。相比于properties来说,yml更加年轻,层级也是更加分明。
properties 和 yml 两者之间区别:
- properties 文件以"." 进行分割,yaml以":"进行分割
- properties 文件通过 “=” 赋值,yml 的数据格式类似json,通过 “=” 赋值,值前面需要加一个空格;yml文件缩进最好用空格
- properties 只支持键值对,yml 配置文件支持列表,短横线 “-” 表示列表
- properties 不保证加载顺序,yml 有先后顺序
properties 文件
Java 中的 properties 文件是一种配置文件,主要用于表达配置信息,文件类型为 “.properties”,格式为文本文件,这种文件以 key=value 格式存储内容。
Properties 类继承自 Hashtable,是线程安全的,多个线程可以共享单个properties 对象而无需进行外部同步。
一般这个文件作为一些参数的存储,代码可以灵活一点。通俗点讲就相当于定义一个变量,在这个文件里面定义这些变量的值,在程序里面可以调用这些变量,好处就是,如果程序中的参数值需要变动,直接修改这个 “.properties” 文件就可以了,不用去修改源代码。
properties 文件案例
application.properties
userinfo.name=example
userinfo.age=25
userinfo.active=true
userinfo.created-date=2023/01/06 11:20:20
userinfo.map.k1=v1
userinfo.map.k2=v2
UserInfo.java
@Data
@ToString
public class UserInfo {
private String name;
private Integer age;
private Boolean active;
private Map<String, Object> map;
private Date createdDate;
private List<String> hobbies;
}
YML 文件
yml 文件是 spring boot 推荐使用的配置文件格式,它相较于 properties 格式文件更强大,比如可以实现 SpEl表达式、松散绑定等。
以空格的缩进程度来控制层级关系。空格的数量并不重要,只要左边空格对齐则视为同一个层级。不能用 tab 代替空格,且大小写敏感。支持字面值、对象和数组三种数据结构,也支持复合结构。
- 字面值:字符串、布尔类型、数值、日期。字符串默认不加引号,单引号会转义字符。日期格式支持 yyyy/MM/dd HH:mm:ss
- 对象:由键值对组成,形如 key: (空格)value 的数据组成。冒号后面的空格是必须要有的,每组键值对占用一行,且缩进的程度要一致,也可以使用行内写法:{k1" v1, ... k2: v2}
- 数组:由形如 - (空格)value 的数据组成。短横线后面的空格必须要有的,每组数据占用一行,且缩进的程度要一致,也可以使用行内写法:[1, 2, ... n]
- 复合结构:上面三种数据结构任意组合
yml 文件案例
application.yaml
UserInfo:
name: example
age: 25
active: true
created-date: 2023/01/06 11:40:00
map: {k1: v1, k2: v2}
hobbies:
- football
- basketball
- ping-pong ball
UserInfo.java
@Data
@ToString
public class UserInfo {
private String name;
private Integer age;
private Boolean active;
private Map<String, Object> map;
private Date createdDate;
private List<String> hobbies;
}
注意
- 字符串可以不加引号,若加双引号则输出特殊字符,若不加或加单引号则转义特殊字符
- 数组类型,短横线后面有空格;对象类型,冒号后面要有空格
- YAML 是以空格缩进的程度来控制层级关系,但不能用 tab 键代替空格,大小写敏感
从配置文件取值
@ConfigurationProperties
这个注解用于从配置文件中取值,支持复杂的数据类型,但是不支持 SPEL 表达式。
该注解中有一个属性 prefix, 用于指定获取配置的前缀,毕竟配置文件中的属性很多,也有很多重名的配置,所以必须用一个前缀来区分下。
该注解可以标注在类上也可以标注在方法上,所以它有两种获取值的方式。
标注在实体类上
这种方式用于从实体类上取值,并且赋值到对应的属性。
/**
* @Component 注入到 IOC 容器中
* @ConfigurationProperties 从配置文件中读取数据
*/
@Data
@ToString
@Component
@ConfigurationProperties(prefix = "userinfo")
public class UserInfo {
private String name;
private Integer age;
private Boolean active;
private Map<String, Object> map;
private Date createdDate;
private List<String> hobbies;
}
在 controller 中注入配置类后进行使用
@RestController
@RequestMapping("demo")
public class DemoController {
@Autowired
private UserInfo userInfo;
@GetMapping("test")
public String demoTest() {
return JSON.toJSONString(userInfo);
}
}
访问对应的 url 之后可以看到在配置文件中配置的信息已经获取到
{"active":true,"age":25,"createdDate":"2023-01-06 11:20:20","hobbies":["football","basketball","ping-pong ball"],"map":{"k1":"v1","k2":"v2"},"name":"example"}
标注在配置类中的方法上
标注在配置类上的方法上,同样是从配置文件中取值赋值到返回值的属性中
/**
* @Bean 将返回的结果注入到 IOC 容器中
* @ConfigurationProperties 从配置文件中取值
* @return
*/
@ConfigurationProperties(prefix = "userinfo")
@Bean
public UserInfo userInfo() {
return new UserInfo();
}
总结:@ConfigurationProperties 注解能够很轻松的从配置文件中取值,优点如下:
- 支持批量的注入属性,只需要指定一个前缀 pregix
- 支持复杂的数据类型,比如 List, Map
- 对属性名匹配的要求较低,比如 user-name, user_name, userName, USER_NAME 都可以取值
- 支持 Java 的 JSR303 数据校验
注意:@ConfigurationProperties 注解仅仅支持从 spring boot 的默认配置文件中取值,比如 application.properties, application.yml, application.yaml
@Value
@Value 这个注解应该是很熟悉了,spring 中从属性取值的注解,支持 SPEL 表达式,不支持复杂的数据类型,比如 List
@RestController
@RequestMapping("demo")
public class DemoController {
@Value("${userinfo.name}")
private String name;
@GetMapping("test")
public String demoTest() {
return JSON.toJSONString(name);
}
}
访问对应的 url 之后可以看到在配置文件中配置的信息已经获取到
"example"
该注解只能是获取到配置的某一条属性值,不能获取到整个对象的属性配置并封装类
从自定义配置文件中取值
spring boot 在启动的时候会自动加载 application.xxx 配置文件,但为了区分,有时候需要自定义一个配置文件,读取这种自定义文件,就需要配合 @propertySource 注解来获取自定义配置文件了,只需要在启动类上标注 @PropertySource 并指定自定义的配置文件即可。
// 文件名称 customConfig.properties
student.name=example
student.age=20
student.birthday=2022/01/06 15:00:00
实体类
@Data
@ToString
@Component
@ConfigurationProperties(prefix = "student")
public class Student {
private String name;
private Integer age;
private Date birthday;
}
启动类
@SpringBootApplication
@PropertySource(value = {"classpath:customConfig.properties"})
public class DemoApplication {
public static void main(String[] args) {
SpringApplication.run(DemoApplication.class, args);
}
}
value 属性是一个数组,可以指定多个配置文件同时引入。
@PropertySource 默认加载 application.properties,不能加载 yml 格式的配置文件,yml 配置文件需要另外处理。
controller
@RestController
@RequestMapping("demo")
public class DemoController {
@Autowired
private Student student;
@GetMapping("test")
public String demoTest() {
return JSON.toJSONString(student);
}
}
启动后访问对应的 url 可以查看到已经能够读取到配置文件属性并对数据进行封装了,能够获取到
{"age":20,"birthday":"2022-01-06 15:00:00","name":"example"}
加载自定义 YML 格式的配置文件
@PropertySource 注解有一个属性 factory,默认是 PropertySourceFactory.class,这个就说用来加载 properties 格式的配置文件,我们可以自定义一个用来加载 yml 格式的配置文件。
public class YmlConfigFactory extends DefaultPropertySourceFactory {
@Override
public PropertySource<?> createPropertySource(String name, EncodedResource resource) throws IOException {
String sourceName = name != null ? name : resource.getResource().getFilename();
if (!resource.getResource().exists()) {
return new PropertiesPropertySource(sourceName, new Properties());
} else if (sourceName.endsWith(".yml") || sourceName.endsWith(".yaml")) {
Properties propertiesFromYaml = loadYml(resource);
return new PropertiesPropertySource(sourceName, propertiesFromYaml);
} else {
return super.createPropertySource(name, resource);
}
}
private Properties loadYml(EncodedResource resource) throws IOException {
YamlPropertiesFactoryBean factory = new YamlPropertiesFactoryBean();
factory.setResources(resource.getResource());
factory.afterPropertiesSet();
return factory.getObject();
}
}
启动类
@SpringBootApplication
@PropertySource(value = {"classpath:customConfig.yml"}, factory = YmlConfigFactory.class)
public class DemoApplication {
public static void main(String[] args) {
SpringApplication.run(DemoApplication.class, args);
}
}
经过测试,可以实现和 properties 自定义配置文件一样的效果。
@PropertySource 指定加载自定义配置文件,默认只能加载 properties 格式,但是可以指定 factory 属性来加载 yml 格式的配置文件。