Spring中静态方法中使用@Resource注解的变量

时间:2021-07-23 13:40:59

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