Spring中靜態方法中使用@Resource註解的變數
阿新 • • 發佈:2018-12-31
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的屬性(依賴注入))。
- 當元件不好歸類的時候,我們可以使用這個註解進行標註。