springboot @value 解决静态属性无法注入问题

微服务框架配置迁移实践
本文分享了将微服务框架中的数据库配置从代码常量迁移到Config服务的实践经验,通过使用Spring框架的@Component和@Value注解,实现了配置的灵活管理和代码的整洁,确保了程序在不修改其他代码的情况下仍能正常运行。

工作中遇到的场景:

采用的微服务框架,之前es和hbase 的空间名和表名等配置写在JAVA Conatant类中,为了调试方便要改放到config服务下。各属性都是 static final,所以面临@Value静态注入问题。

操作步骤

  1. 创建一个类,+注解component,被spring托管
  2. 声明静态属性和get,set方法。注意set方法不是satic,get方法和属性有static关键字。
  3. 在set方法加入@Value()注解

例子就是下面段代码:

@Component
public class DBNameConfig {

	private static String ES_DB_NAME;

	private static String HB_DB_NAMESPACE;

	public static String getES_DB_NAME() {
		return ES_DB_NAME;
	}

	@Value("${ES_DB_NAME}")
	public void setES_DB_NAME(String eS_DB_NAME) {
		ES_DB_NAME = eS_DB_NAME;
	}

	public static String getHB_DB_NAMESPACE() {
		return HB_DB_NAMESPACE;
	}

	@Value("${HB_DB_NAMESPACE}")
	public void setHB_DB_NAMESPACE(String hB_DB_NAMESPACE) {
		HB_DB_NAMESPACE = hB_DB_NAMESPACE;
	}

}

调用的代码

@Component
public class Constant{
    @Autowired
	DBNameConfig dBNameConfig;
    ... #省略部分代码
    public static final String ES_DB_NAME = DBNameConfig.getES_DB_NAME();
    public static final String HB_DB_NAMESPACE = DBNameConfig.getHB_DB_NAMESPACE();
    ...
    
}

这样就不用改其他代码,配置移到Config服务下,程序也能正常运行。

还没想清楚的问题:读书的时候记得java的加载顺序是先静态,这里的属性还加了final。难道加载顺序是先spring 后static ,感觉不合理…

在 Spring Boot 中,`@Value` 注解主要用于从配置文件中注入属性值到 Bean 中。虽然 `@Value` 本身并不直接支持将 JSON 格式的配置直接转换为 Java 对象,但可以通过结合 SpEL(Spring Expression Language)表达式和自定义解析方法实现这一功能。 ### 使用 SpEL 表达式结合自定义解析器 可以通过 SpEL 调用静态方法来解析 JSON 字符串。例如,可以使用 `Jackson` 或 `Gson` 等库来实现 JSON 字符串的解析,并通过 `@Value` 注解调用该方法。 ```java @Value("#{T(com.example.utils.JsonParser).parseJsonToMap('${my.json.config}')}") private Map<String, Object> configMap; ``` 在上面的示例中,`${my.json.config}` 是配置文件中的 JSON 字符串,`JsonParser.parseJsonToMap()` 是一个静态方法,用于将 JSON 字符串解析为 `Map<String, Object>`。 对应的解析工具类如下: ```java public class JsonParser { private static final ObjectMapper objectMapper = new ObjectMapper(); public static Map<String, Object> parseJsonToMap(String json) { try { return objectMapper.readValue(json, new TypeReference<Map<String, Object>>() {}); } catch (Exception e) { throw new RuntimeException("Failed to parse JSON to Map", e); } } } ``` ### 使用 `@ConfigurationProperties` 替代方案 对于更复杂的对象结构,推荐使用 `@ConfigurationProperties` 注解。它支持将配置文件中的属性直接绑定到 Java 对象上,且天然支持嵌套结构和类型安全。 ```java @ConfigurationProperties(prefix = "my.json") public class MyConfig { private Map<String, Object> config; // getter and setter } ``` 然后在配置文件中定义: ```yaml my: json: config: key1: value1 key2: value2 ``` 这种方式比 `@Value` 更加灵活和强大,尤其适用于需要映射复杂结构的场景。 ### 支持默认值 在使用 `@Value` 时,还可以通过 `:` 指定默认值,以避免配置缺失时出现异常: ```java @Value("#{T(com.example.utils.JsonParser).parseJsonToMap('${my.json.config:{"defaultKey":"defaultValue"}}')}") private Map<String, Object> configMap; ``` ### 注意事项 - `@Value` 不支持嵌套的 JSON 对象直接绑定到 Java 对象,必须通过自定义解析器进行处理。 - 在使用 `@ConfigurationProperties` 时,建议启用配置属性验证以确保配置的完整性。 - 对于大型项目或复杂配置,优先考虑使用 `@ConfigurationProperties` 来提升可维护性和类型安全性。 ---
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值