Java Spring

Spring 中获取配置的三种方式

  1. 通过 @Value 方式动态获取单个配置
  2. 通过 @ConfigurationProperties + 前缀方式批量获取配置
  3. 通过 Environment 动态获取单个配置

    通过 @Value 动态获取单个配置

    作用

  4. 可修饰到任一变量获取,使用较灵活

    优点

  5. 使用简单,且使用关联的链路较短

    缺点

  6. 配置名不能被有效枚举到

  7. 每一个配置的使用都需重新定义,使用较为麻烦
  8. 项目强依赖配置的定义,配置不存在则会导致项目无法启动

    使用场景

  9. 项目强依赖该配置的加载,想要从源头避免因配置缺失导致的未知问题

  10. 只想使用少数几个配置

    代码示例

    1. @Configuration
    2. public class ConfigByValueAnnotation {
    3. @Value("${server.port}")
    4. private String serverPort;
    5. public String getServerPort() {
    6. return serverPort;
    7. }
    8. }

    测试代码

    1. @DisplayName("multipart get config")
    2. @SpringBootTest
    3. public class MultipartGetConfigTest {
    4. private static final Logger log = LoggerFactory.getLogger(MultipartGetConfigTest.class);
    5. @Autowired
    6. private ConfigByValueAnnotation configByValueAnnotation;
    7. @Test
    8. public void getByValueAnnotation(){
    9. log.info("get by @Value, value: {}", configByValueAnnotation.getServerPort());
    10. }
    11. }

    测试结果

    1. org.spring.demo.MultipartGetConfigTest : get by @Value, value: 7100

    通过 @ConfigurationProperties + 前缀方式批量获取

    作用

  11. 用于配置类的修饰或批量配置的获取

    优点

  12. 使用配置只需确定 key 的前缀即能使用,有利于批量获取场景的使用

  13. 因采用前缀匹配,所以在使用新的相同前缀 key 的配置时无需改动代码

    缺点

  14. 使用复杂,需定义配置类或者手动创建 bean 后引入使用

  15. 增加新的前缀相同 key 时可能会引入不稳定因素

    使用场景

  16. 需要同时使用多前缀相同 key 的配置

  17. 期望增加新配置但不修改代码的 properties 注入

    代码示例

    ```java @Component @ConfigurationProperties(prefix = “server”, ignoreInvalidFields = true) public class ConfigByConfigurationProperties {

    private Integer port;

    public Integer getPort() {

    1. return port;

    }

    public ConfigByConfigurationProperties setPort(Integer port) {

    1. this.port = port;
    2. return this;

    } }

@Configuration public class ConfigByConfigurationPropertiesV2 {

  1. @Bean("configByValueAnnotationV2")
  2. @ConfigurationProperties(prefix = "server2")
  3. public Properties properties(){
  4. return new Properties();
  5. }

}

  1. <a name="K8TeD"></a>
  2. #### 测试代码
  3. ```java
  4. @DisplayName("multipart get config")
  5. @SpringBootTest
  6. public class MultipartGetConfigTest {
  7. private static final Logger log = LoggerFactory.getLogger(MultipartGetConfigTest.class);
  8. @Autowired
  9. private ConfigByConfigurationProperties configByConfigurationProperties;
  10. @Autowired
  11. @Qualifier("configByValueAnnotationV2")
  12. private Properties properties;
  13. @Test
  14. public void getByConfigurationProperties(){
  15. log.info("get by @ConfigurationProperties, value: {}", configByConfigurationProperties.getPort());
  16. log.info("get by @ConfigurationProperties and manual create bean, value: {}", properties.getProperty("port"));
  17. }
  18. }

测试结果

  1. org.spring.demo.MultipartGetConfigTest : get by @ConfigurationProperties, value: 7100
  2. org.spring.demo.MultipartGetConfigTest : get by @ConfigurationProperties and manual create bean, value: 7100

通过 Environment 动态获取单个配置

作用

  1. 用于动态在程序代码中获取配置,而配置 key 不需提前定义

    优点

  2. 获取的配置的 key 可不提前定义,程序灵活性高

  3. 配置 key 可使用枚举统一放置与管理

    缺点

  4. 使用较复杂,需继承 Environment 接口形成工具类进行获取

  5. 获取 key 对应的枚举与 key 定义分离,value 获取链路较长

    使用场景

  6. 只需使用少量的配置

  7. 获取配置的 key 无提前定义,需要根据对配置的有无进行灵活使用

    代码示例

    ```java @Component public class ConfigByEnvironment implements EnvironmentAware {

    private static final Logger log = LoggerFactory.getLogger(ConfigByEnvironment.class);

    private Environment environment;

    public Optional get(String configKey){

    1. String config = environment.getProperty(configKey);
    2. return Objects.isNull(config) ? Optional.empty() : Optional.of(config);

    }

    public void get(String configKey, Consumer consumer){

    1. Optional<String> config = get(configKey);
    2. if(!config.isPresent()){
    3. log.warn("application config, get config by key fail, key: {}", configKey);
    4. }
    5. config.ifPresent(consumer);

    }

    @Override public void setEnvironment(@NonNull Environment environment) {

    1. this.environment = environment;

    } }

public enum ConfigByEnvironmentKey {

  1. SERVER_PORT("server.port", "server port");
  2. private String key;
  3. private String description;
  4. ConfigByEnvironmentKey(String key, String description) {
  5. this.key = key;
  6. this.description = description;
  7. }
  8. public String getKey() {
  9. return key;
  10. }
  11. public String getDescription() {
  12. return description;
  13. }

}

  1. <a name="sVcvt"></a>
  2. #### 测试代码
  3. ```java
  4. @DisplayName("multipart get config")
  5. @SpringBootTest
  6. public class MultipartGetConfigTest {
  7. private static final Logger log = LoggerFactory.getLogger(MultipartGetConfigTest.class);
  8. @Autowired
  9. private ConfigByEnvironment configByEnvironment;
  10. @Test
  11. public void getByEnvironment(){
  12. configByEnvironment.get(ConfigByEnvironmentKey.SERVER_PORT.getKey()).ifPresent(value -> log.info("get by environment, value: {}", value));
  13. }
  14. }

测试结果

  1. org.spring.demo.MultipartGetConfigTest : get by environment, value: 7100

总结

Spring 获取配置的三种方式 - 图1