開發中,有些時候可能會工具類的靜態方法,而這個靜態方法中又使用到了@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); }
下面簡單介紹一下相關注解
- 1.@Component
- 泛指組件,當組件不好歸類的時候,我們可以使用這個注解進行標注。
2.@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)進行匹配,如果匹配則自動裝配;
---------------------
3. @PostConstruct
- 在方法上加上注解@PostConstruct,這個方法就會在Bean初始化之后被Spring容器執行(注:Bean初始化包括,實例化Bean,並裝配Bean的屬性(依賴注入))。