今天在写项目的时候,因项目里的某个key是每个人都不一样,所以想到了在application.yml里配置,随后使用@Value注入,不同人使用就只需要创建每个人的application文件就行了。

大概就是下面这样:

@Value("${localhost-key}")
    private static String key;

yml里:

localhost-key: abcd

结果不知道为什么,@Value注入一直为空??
很疑惑,检查了一下@Value并没有导错包,@Value("${localhost-key}")也没有写错,再仔细检查了一下Utils,发现没加@Component!!!

有点郁闷的,赶紧把@Component加上了:

@Component
public class FileUtils {

    @Value("${localhost-key}")
    private static String key;
}

再一运行,发现居然还是null
心里很疑惑,到底哪里出了问题,运行没有问题,也没有报错,感觉就是@Value的问题了。

一番查找资料,又问了问学校的老师,终于明白:
@Value不能直接注入值给静态属性,Spring 不允许(或者说不支持)把值注入到静态变量中

========================================================================
原因如下:

静态变量是一个类的属性,而不是对象的属性,但是Spring的依赖注入是基于对象层面上的。

而使用静态变量扩大了静态方法的使用范围,Spring是不推荐使用静态方法的,依赖注入的目的就是想让容器去产生一个对象的实例,然后在整个生命周期中使用他们。

而一旦使用静态方法,就不会去产生这个类的实例,这会让测试变得更加困难。同时也不能为一个给定的类,依靠注入方式去产生多个具有不同的依赖环境的实例。

这种静态域是隐含共享的,并且是一种全局状态,Spring同样不推荐这样去做。

========================================================================

问题倒是找到了,但是有没有解决的办法呢?又是一番寻找之后
解决办法:
Spring支持set方法注入,我们可以利用非静态Set方法注入静态变量,并且使用@Value的类必须交给spring进行管理(即类上添加@Compent、@RestController、@Service…)

@Component
public class FileUtils {
    	private static String key;
   
    	@Value("${localhost-key}")
    	public void setKey(String key) {    //注意这里的set方法不能是静态的
        	FileUtils.key = key;  
    	}
  }

再一运行,果然值就赋上了。
问题解决~