1. 前言
在Spring Boot專案中我們經常需要讀取application.yml
配置檔案的自定義配置,今天就來羅列一下從yaml
讀取配置檔案的一些常用手段和方法。
2. 使用@Value註解
首先,會想到使用@Value
註解,該註解只能去解析yaml
檔案中的簡單型別,並繫結到物件屬性中去。
felord:
phone: 182******32
def:
name: 碼農小胖哥
blog: felord.cn
we-chat: MSW_623
dev:
name: 碼農小胖哥
blog: felord.cn
we-chat: MSW_623
type: JUEJIN
對於上面的yaml
配置,如果我們使用@Value
註解的話,冒號後面直接有值的key
才能正確注入對應的值。例如felord.phone
我們可以通過@Value
獲取,但是felord.def
不行,因為felord.def
後面沒有直接的值,它還有下一級選項。另外@Value
不支援yaml
鬆散繫結語法,也就是說felord.def.weChat
獲取不到felord.def.we-chat
的值。
@Value
是通過使用Spring的SpEL
表示式來獲取對應的值的:
// 獲取 yaml 中 felord.phone的值 並提供預設值 UNKNOWN
@Value("${felord.phone:UNKNOWN}")
private String phone;
@Value
的使用場景是隻需要獲取配置檔案中的某項值的情況下,如果我們需要將一個系列的值進行繫結注入就建議使用複雜物件的形式進行注入了。
3. 使用@ConfigurationProperties註解
@ConfigurationProperties
註解提供了我們將多個配置選項注入複雜物件的能力。它要求我們指定配置的共同字首。比如我們要繫結felord.def
下的所有配置項:
package cn.felord.yaml.properties;
import lombok.Data;
import org.springframework.boot.context.properties.ConfigurationProperties;
import static cn.felord.yaml.properties.FelordDefProperties.PREFIX;
/**
* @author felord.cn
*/
@Data
@ConfigurationProperties(PREFIX)
public class FelordDefProperties {
static final String PREFIX = "felord.def";
private String name;
private String blog;
private String weChat;
}
我們注意到我們可以使用
we-chat
的值,因為這種形式支援從駝峰camel-case
到短橫分隔命名kebab-case
的自動轉換。
如果我們使用@ConfigurationProperties
的話建議配置類命名字尾為Properties
,比如Redis的字尾就是RedisProperties
,RabbitMQ的為RabbitProperties
。
另外我們如果想進行巢狀的話可以藉助於@NestedConfigurationProperty
註解實現。也可以藉助於內部類。這裡用內部類實現將開頭yaml
中所有的屬性進行注入:
package cn.felord.yaml.properties;
import lombok.Data;
import org.springframework.boot.context.properties.ConfigurationProperties;
import static cn.felord.yaml.properties.FelordProperties.PREFIX;
/**
* 內部類和列舉配置.
*
* @author felord.cn
*/
@Data
@ConfigurationProperties(PREFIX)
public class FelordProperties {
static final String PREFIX = "felord";
private Def def;
private Dev dev;
private Type type;
@Data
public static class Def {
private String name;
private String blog;
private String weChat;
}
@Data
public static class Dev {
private String name;
private String blog;
private String weChat;
}
public enum Type {
JUEJIN,
SF,
OSC,
CSDN
}
}
單獨使用@ConfigurationProperties
的話依然無法直接使用配置物件FelordDefProperties
,因為它並沒有被註冊為Spring Bean。我們可以通過兩種方式來使得它生效。
3.1 顯式注入Spring IoC
你可以使用@Component
、@Configuration
等註解將FelordDefProperties
注入Spring IoC使之生效。
package cn.felord.yaml.properties;
import lombok.Data;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.stereotype.Component;
import static cn.felord.yaml.properties.FelordDefProperties.PREFIX;
/**
* 顯式注入Spring IoC
* @author felord.cn
*/
@Data
@Component
@ConfigurationProperties(PREFIX)
public class FelordDefProperties {
static final String PREFIX = "felord.def";
private String name;
private String blog;
private String weChat;
}
3.2 使用@EnableConfigurationProperties註冊
我們還可以使用註解@EnableConfigurationProperties
進行註冊,這樣就不需要顯式宣告配置類為Spring Bean了。
package cn.felord.yaml.configuration;
import cn.felord.yaml.properties.FelordDevProperties;
import org.springframework.boot.context.properties.EnableConfigurationProperties;
import org.springframework.context.annotation.Configuration;
/**
* 使用 {@link EnableConfigurationProperties} 註冊 {@link FelordDevProperties}使之生效
* @author felord.cn
*/
@EnableConfigurationProperties({FelordDevProperties.class})
@Configuration
public class FelordConfiguration {
}
該註解需要顯式的註冊對應的配置類。
3.3 使用@ConfigurationPropertiesScan掃描
在Spring Boot 2.2.0.RELEASE中提供了一個掃描註解@ConfigurationPropertiesScan
。它可以掃描特定包下所有的被@ConfigurationProperties
標記的配置類,並將它們進行IoC注入。
package cn.felord.yaml;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.boot.context.properties.ConfigurationPropertiesScan;
import org.springframework.boot.context.properties.EnableConfigurationProperties;
/**
* {@link ConfigurationPropertiesScan} 同 {@link EnableConfigurationProperties} 二選一
*
* @see cn.felord.yaml.configuration.FelordConfiguration
* @author felord.cn
*/
@ConfigurationPropertiesScan
@SpringBootApplication
public class SpringBootYamlApplication {
public static void main(String[] args) {
SpringApplication.run(SpringBootYamlApplication.class, args);
}
}
這非常適合自動注入和批量注入配置類的場景,但是有版本限制,必須在2.2.0及以上。
3.4 Environment
Spring Boot專案的話也可以通過org.springframework.core.env.Environment
提供的getProperty(String key)
來獲取,一般並不是很常用。
4. 總結
日常開發中單個屬性推薦使用@Value
,如果同一組屬性為多個則推薦@ConfigurationProperties
。需要補充一點的是@ConfigurationProperties
還支援使用JSR303進行屬性校驗。多多關注:碼農小胖哥 獲取更多的技術乾貨。相關的demo 可通過公眾號回覆yaml
獲取。
關注公眾號:Felordcn 獲取更多資訊