开发中,有些时候可能会工具类的静态方法,而这个静态方法中又使用到了@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的属性(依赖注入))。