Spring 注入static属性值
本文介绍Spring中如何从属性文件给static字段注入值。实际应用中一些工具类中static属性值需读取配置文件,实现该功能可以让工具类提供静态方法更易使用。
1. 问题
首先在属性文件中定义属性:
1
|
name = Inject a value to a static field |
然后给实例变量注入值,通常在字段上使用@Value注解:
1
2
|
@Value ( "${name}" ) private String name; |
但在static字段上应用是,会发现其值为null,注入失败:
1
2
|
@Value ( "${name}" ) private static String NAME_NULL; |
这是因为Spring不支持在static字段上使用@Value注解。
2. 解决方案
Spring @Value注解可以在方法上使用,在加载所有Spring配置和bean后,Spring上下文将调用它。方法有多个参数,那么每个参数值都为方法注解对应的值,如果需要参数获取不同的值,可以在参数上增加注解:
1
2
3
4
5
|
@Value ( "Test" ) public void printValues(String s, String v){} //both 's' and 'v' values will be 'Test' @Value ( "Test" ) public void printValues(String s, @Value ( "Data" ) String v){} // s=Test, v=Data |
有了上面的知识,我们可以修改代码为:
1
2
3
4
5
6
7
8
9
|
public class PropertyUtils { @Value ( "${name1}" ) private String name; private static String NAME_STATIC; @Value ( "${name2}" ) public void setNameStatic(String name){ PropertyController.NAME_STATIC = name; } } |
这回通过方法成功给static变量NAME_STATIC赋值。
Spring依赖注入static静态变量相关问题
1.Spring不支持依赖注入static静态变量
在springframework里,我们不能@Autowired一个静态变量,使之成为一个spring bean,例如下面这样:
1
2
|
@Autowired private static YourClass yourClass; |
可以试一下,yourClass在这种状态下不能够被依赖注入,会抛出运行时异常java.lang.NullPointerException,为什么呢?静态变量/类变量不是对象的属性,而是一个类的属性,spring则是基于对象层面上的依赖注入.
而使用静态变量/类变量扩大了静态方法的使用范围.静态方法在spring是不推荐使用的.依赖注入的主要目的,是让容器去产生一个对象的实例,然后在整个生命周期中使用他们,同时也让testing工作更加容易.
一旦你使用静态方法,就不再需要去产生这个类的实例,这会让testing变得更加困难,同时你也不能为一个给定的类,依靠注入方式去产生多个具有不同的依赖环境的实例.这种static field是隐含共享的,并且是一种global全局状态,spring同样不推荐这样去做.
2.Spring如何给静态变量注入值
spring 不允许/不支持把值注入到静态变量中,如:
1
2
3
4
5
6
7
|
import org.springframework.beans.factory.annotation.Value; import org.springframework.stereotype.Component; @Component public class GlobalValue { @Value ( "${mongodb.db}" ) public static String DATABASE; } |
如果你获取GlobalValue.DATABASE,会得到null
1
|
GlobalValue.DATABASE = null |
那我们如何解决这个问题呢。
好在spring支持set方法注入,我们可以利用非静态setter 方法注入静态变量。如:
1
2
3
4
5
6
7
8
9
10
|
import org.springframework.beans.factory.annotation.Value; import org.springframework.stereotype.Component; @Component public class GlobalValue { public static String DATABASE; @Value ( "${mongodb.db}" ) public void setDatabase(String db) { DATABASE = db; } } |
输出:
GlobalValue.DATABASE = "mongodb database name"
3.Spring静态注入的三种方式
(说明:MongoFileOperationUtil是自己封装的一个Mongodb文件读写工具类,里面需要依赖AdvancedDatastore对象实例,dsForRW用来获取Mongodb数据源)
在springframework里,我们不能@Autowired一个静态变量,使之成为一个spring bean,例如下面这种方式:
1
2
|
@Autowired private static AdvancedDatastore dsForRW; |
可以试一下,dsForRW在这种状态下不能够被依赖注入,会抛出运行时异常java.lang.NullPointerException,为什么呢?静态变量/类变量不是对象的属性,而是一个类的属性,spring则是基于对象层面上的依赖注入。
但是自己比较喜欢封装工具类,并通过@Component注解成功能组件,但是功能组件中的方法一般都是静态方法,静态方法只能调用静态成员变量,于是就有了下面的问题。封有的时候封装功能组件会需要底层的service注入,怎么办呢?
去网上搜了下解决办法,简单总结一下几种实现方式
1.xml方式实现
1
2
3
|
< bean id = "mongoFileOperationUtil" class = "com.*.*.MongoFileOperationUtil" init-method = "init" > < property name = "dsForRW" ref = "dsForRW" /> </ bean > |
1
2
3
4
5
6
7
8
|
public class MongoFileOperationUtil { private static AdvancedDatastore dsForRW; private static MongoFileOperationUtil mongoFileOperationUtil; public void init() { mongoFileOperationUtil = this ; mongoFileOperationUtil.dsForRW = this .dsForRW; } } |
这种方式适合基于XML配置的WEB项目;
2.@PostConstruct方式实现
1
2
3
4
5
6
7
8
9
10
11
12
13
|
import org.mongodb.morphia.AdvancedDatastore; import org.springframework.beans.factory.annotation.Autowired; @Component public class MongoFileOperationUtil { @Autowired private static AdvancedDatastore dsForRW; private static MongoFileOperationUtil mongoFileOperationUtil; @PostConstruct public void init() { mongoFileOperationUtil = this ; mongoFileOperationUtil.dsForRW = this .dsForRW; } } |
@PostConstruct 注解的方法在加载类的构造函数之后执行,也就是在加载了构造函数之后,执行init方法;(@PreDestroy 注解定义容器销毁之前的所做的操作)
这种方式和在xml中配置 init-method和 destory-method方法差不多,定义spring 容器在初始化bean 和容器销毁之前的所做的操作;
3.set方法上添加@Autowired注解,类定义上添加@Component注解
1
2
3
4
5
6
7
8
9
10
11
|
import org.mongodb.morphia.AdvancedDatastore; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.stereotype.Component; @Component public class MongoFileOperationUtil { private static AdvancedDatastore dsForRW; @Autowired public void setDatastore(AdvancedDatastore dsForRW) { MongoFileOperationUtil.dsForRW = dsForRW; } } |
首先Spring要能扫描到AdvancedDatastore的bean,然后通过setter方法注入;
然后注意:成员变量上不需要再添加@Autowired注解;
原文链接:https://blog.csdn.net/neweastsun/article/details/105425403