解决SpringBoot @Value注解静态属性无法注入问题

本文介绍了在SpringBoot中遇到@Value注解无法注入静态属性的问题及其两种解决方案。方法一是通过setter方法设置静态变量,方法二是利用@PostConstruct初始化方法。这两种方式都能确保静态属性在应用启动时被正确注入。


最近在项目中遇到一个问题:
在给static修饰的变量做@Value注入时,发现变量值为null,发现@Value只能给普通变量做值注入,无法为static属性赋值。

@Data
@Component
public class ApiConfig {
    @Value("${openapi.appKey:test21c454457f28a2c31}")
    private String appKey;
}

解决方案一:使用 set 方法

抽取出配置类ApiConfig,并添加@Component注解

@Component
public class ApiConfig {
    /**
     * 设置是否沙箱环境
     */
    public static boolean IS_SANDBOX;
    /**
     * 设置APP_ID
     */
    public static String APP_ID;

    @Value("${openapi.isSandbox:true}")
    public void setIS_SANDBOX(boolean isSandbox) {
        IS_SANDBOX = isSandbox;
    }

    @Value("${openapi.appId:143771897797159}")
    public void setAPP_ID(String appId) {
        APP_ID = appId;
    }

    public static boolean isIsSandbox() {
        return IS_SANDBOX;
    }

    public static String getAppId() {
        return APP_ID;
    }
}

单元测试

@Autowired
ApiConfig apiConfig;

@Test
void testApiConfig(){
     System.out.println(eleApiConfig.getAppId());
     System.out.println("=================================");
     System.out.println(EleApiConfig.getAppId());
}

解决方案二:使用 @PostConstruct 方法

@Component
public class ApiConfig {
    /**
     * 设置是否沙箱环境
     */
    public static boolean IS_SANDBOX;
    
    @Value("${openapi.isSandbox:true}")
    public boolean is_sandbox;
    
    /**
     * 设置APP_ID
     */
    public static String APP_ID;
    @Value("${openapi.appId:143771897797159}")
    public String appId;
    
    public static boolean isIsSandbox() {
        return IS_SANDBOX;
    }

    public static String getAppId() {
        return APP_ID;
    }
  
    @PostConstruct
    public void init() {
    	IS_SANDBOX = is_sandbox;
    	APP_ID = appId;
   }
}

参考文章
springboot @value 解决静态属性无法注入问题
SpringBoot使用@Value给静态变量注入值
springboot 使用 @Value 注解给静态变量赋值,实现配置全局化
SpringBoot中静态变量注入方案,一网打尽

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` 来提升可维护性和类型安全性。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值