Spring 依賴注入 static 靜態變量相關問題
1.Spring 不支持依賴注入 static 靜態變量
在 springframework 里, 我們不能 @Autowired 一個靜態變量, 使之成為一個 spring bean, 例如下面這樣:
@Autowired
private static YourClass yourClass;
可以試一下, yourClass 在這種狀態下不能夠被依賴注入, 會拋出運行時異常 java.lang.NullPointerException, 為什么呢? 靜態變量 / 類變量不是對象的屬性, 而是一個類的屬性, spring 則是基於對象層面上的依賴注入.
而使用靜態變量 / 類變量擴大了靜態方法的使用范圍. 靜態方法在 spring 是不推薦使用的. 依賴注入的主要目的, 是讓容器去產生一個對象的實例, 然后在整個生命周期中使用他們, 同時也讓 testing 工作更加容易.
一旦你使用靜態方法, 就不再需要去產生這個類的實例, 這會讓 testing 變得更加困難, 同時你也不能為一個給定的類, 依靠注入方式去產生多個具有不同的依賴環境的實例. 這種 static field 是隱含共享的, 並且是一種 global 全局狀態, spring 同樣不推薦這樣去做.
2.Spring 如何給靜態變量注入值
spring 不允許 / 不支持把值注入到靜態變量中,如:
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
GlobalValue.DATABASE = null
那我們如何解決這個問題呢。
好在 spring 支持 set 方法注入,我們可以利用非靜態 setter 方法注入靜態變量。如:
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 靜態注入的三種方式
Spring 靜態注入的三種方式:
(說明:MongoFileOperationUtil 是自己封裝的一個 Mongodb 文件讀寫工具類,里面需要依賴 AdvancedDatastore 對象實例,dsForRW 用來獲取 Mongodb 數據源)
在 springframework 里,我們不能 @Autowired 一個靜態變量, 使之成為一個 spring bean,例如下面這種方式:
@Autowired
private static AdvancedDatastore dsForRW;
可以試一下,dsForRW 在這種狀態下不能夠被依賴注入,會拋出運行時異常 java.lang.NullPointerException,為什么呢? 靜態變量 / 類變量不是對象的屬性, 而是一個類的屬性, spring 則是基於對象層面上的依賴注入。
但是自己比較喜歡封裝工具類,並通過 @Component 注解成功能組件,但是功能組件中的方法一般都是靜態方法,靜態方法只能調用靜態成員變量,於是就有了下面的問題。封有的時候封裝功能組件會需要底層的 service 注入,怎么辦呢?
去網上搜了下解決辦法,簡單總結一下幾種實現方式;
1.xml 方式實現:
<bean init-method="init">
<property name="dsForRW" ref="dsForRW"/>
</bean>
public class MongoFileOperationUtil {
private static AdvancedDatastore dsForRW;
private static MongoFileOperationUtil mongoFileOperationUtil;
public void init() {
mongoFileOperationUtil = this;
mongoFileOperationUtil.dsForRW = this.dsForRW;
}
}
這種方式適合基於 XML 配置的 WEB 項目;
2.@PostConstruct 方式實現;
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 注解;
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 注解;
本文借鑒多篇博客:
http://blog.csdn.net/randyfeng007/article/details/45330793
http://blog.csdn.net/doctor_who2004/article/details/50532169
http://blog.csdn.net/chen1403876161/article/details/53644024