Spring框架中使用靜態注入
開發中,有些時候可能會工具類的靜態方法,而這個靜態方法中又使用到了@Resource註解後的變量。如果要直接使用 Utils.staticMethod(),項目會報異常;如果不直接使用,還要先 new Utils().staticMethod() 吧啦吧啦一大堆!對於一個強迫症碼農不能忍! 那麼,問題來了…
- 例如下面代碼:
/**
* @Description 業務開關工具類
* @Author ZF
* @Date 2017/8/24 15:53
*/
public class SwitchUtil {
private static MyLogger log = MyLogger.getLogger(SwitchUtil.class);
@Resource
private SysConfigManager sysConfigManager;
/**
* 這是一個靜態方法,這個方法中使用到了sysConfigManager這個由@Resource註解的變量
* 看似這樣就可以使用了,其實不行,項目會報錯。
*/
public static boolean getSwitch(String code) {
String switchName = sysConfigManager.getSysConfigByCode("switch").getName();
JSONObject jsonObject = JSONObject.fromObject(switchName);
return jsonObject.getBoolean(code);
}
上面的代碼啓動報如下異常:
java.lang.IllegalStateException: @Resource annotation is not supported on static fields
下面來看看該如何修改
- 修改後的代碼
/**
* @Description 業務開關工具類
* @Author ZF
* @Date 2017/8/24 15:53
*/
@Component
public class SwitchUtil {
private static MyLogger log = MyLogger.getLogger(SwitchUtil.class);
@Resource
private SysConfigManager sysConfigManager;
// 維護一個本類的靜態變量
public static SwitchUtil switchUtil;
// 初始化的時候,將本類中的sysConfigManager賦值給靜態的本類變量
@PostConstruct
public void init() {
switchUtil = this;
switchUtil.sysConfigManager = this.sysConfigManager;
}
/**
* 通過使用本類中維護的靜態變量來使用sysConfigManager
*/
public static boolean getSwitch(String code) {
String switchName = switchUtil.sysConfigManager.getSysConfigByCode("switch").getName();
JSONObject jsonObject = JSONObject.fromObject(switchName);
return jsonObject.getBoolean(code);
}
相關說明在代碼註釋中
下面簡單介紹一下相關注解
@Component
泛指組件,當組件不好歸類的時候,我們可以使用這個註解進行標註。
@Resource
- Spring 不但支持自己定義的@Autowired註解,還支持幾個由JSR-250規範定義的註解,它們分別是@Resource、@PostConstruct以及@PreDestroy。
- @Resource的作用相當於@Autowired,只不過@Autowired按byType自動注入,而@Resource默認按 byName自動注入罷了。@Resource有兩個屬性是比較重要的,分別是name和type,Spring將@Resource註解的name屬性解析爲bean的名字,而type屬性則解析爲bean的類型。所以如果使用name屬性,則使用byName的自動注入策略,而使用type屬性時則使用byType自動注入策略。如果既不指定name也不指定type屬性,這時將通過反射機制使用byName自動注入策略。
- @Resource裝配順序:
- 如果同時指定了name和type,則從Spring上下文中找到唯一匹配的bean進行裝配,找不到則拋出異常
- 如果指定了name,則從上下文中查找名稱(id)匹配的bean進行裝配,找不到則拋出異常
- 如果指定了type,則從上下文中找到類型匹配的唯一bean進行裝配,找不到或者找到多個,都會拋出異常
- 如果既沒有指定name,又沒有指定type,則自動按照byName方式進行裝配(見2);如果沒有匹配,則回退爲一個原始類型(UserDao)進行匹配,如果匹配則自動裝配;
@PostConstruct
- 在方法上加上註解@PostConstruct,這個方法就會在Bean初始化之後被Spring容器執行(注:Bean初始化包括,實例化Bean,並裝配Bean的屬性(依賴注入))。
- 當組件不好歸類的時候,我們可以使用這個註解進行標註。