Spring不能直接@autowired注入Static变量问题和解决方案




Spring不能直接@autowired注入Static变量问题和解决方案

@autowired注入Static变量会报java.lang.NullPointerException: null异常。

原因:

当类加载器加载静态变量时,Spring的上下文环境还没有被加载。

这是因为初始化类的加载升序导致的,程序启动时会加载根路径下所有的类,不管这个类是否会用到都会去加载;会先初始化静态变量和执行静态代码块,这时候无法创建对象,而@autowired是要注入一个对象

查看AutowiredAnnotationBeanPostProcessor类源码:

List<InjectedElement> currElements = new ArrayList();
ReflectionUtils.doWithLocalFields(targetClass, (field) -> {
    AnnotationAttributes ann = this.findAutowiredAnnotation(field);
    if (ann != null) {
        if (Modifier.isStatic(field.getModifiers())) {
            if (this.logger.isInfoEnabled()) {
                this.logger.info("Autowired annotation is not supported on static fields: " + field);
            }

            return;
        }

        boolean required = this.determineRequiredStatus(ann);
        currElements.add(new AutowiredAnnotationBeanPostProcessor.AutowiredFieldElement(field, required));
    }

});

注入的最真实原因:扫描Class类需要注入的元数据的时候,直接选择忽略掉了static成员(包括属性和方法)。

一些思考:

spring有注入机制,我们不应该或者是不推荐使用静态的属性或者方法,
因为一旦你使用静态方法,就不再需要去产生这个类的实例,自然也就不需要注入了,同时你也不能为一个给定的类,依靠注入方式去产生多个具有不同的依赖环境的实例,这种静态属性是隐含共享的,并且是一种global全局状态。



导致的一些的问题:

有些情况,我们还是需要使用static方法的,比如在工具类中使用static方法,这时我们就需要使用static属性,而这些属性又必须注入。



解决方案一:使用@PostConstruct注解

@PostConstruct用来修饰一个非静态的void方法,当bean创建完成的时候,会后置执行@PostConstruct修饰的方法,并且只运行一次。

可以将static变量延迟初始化,在能够注入后进行初始化。

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.stereotype.Component;

import javax.annotation.PostConstruct;

@Component
public class RedisUtil {

    private static RedisTemplate<Object, Object> redisTemplates;

    @Autowired
    private RedisTemplate<Object, Object> redisTemplate;

    @PostConstruct
    public void initialize() {
        redisTemplates = this.redisTemplate;
    }

    /**
     * 添加元素
     *
     * @param key
     * @param value
     */
    public static void set(Object key, Object value) {

        if (key == null || value == null) {
            return;
        }
        redisTemplates.opsForValue().set(key, value);
    }
}

存在的问题:得保证在bean初始化之前,这个工具类的static方法不会被调用。



解决方案二:给静态变量设置Setter方法,在Setter方法上使用@Autowired注解

@RunWith(SpringRunner.class)
@SpringBootTest
public class RedisTest {

    private static ResourceService resourceService;
    
    @Autowired
    public void setService(ResourceService resourceService)  {
        RedisTest.resourceService = resourceService;
    }

    @Test
    public void test(){
        System.out.println(resourceService);
    }
}

之前我一直以为这种方式只能使用 setxxx才可以成功的注入属性 ,那其实是我理解错了,这个方法可以任意的命名

其实跟构造方法一样,这个方法也可以同时有多个传参,Spring 将会把这些属性一起注入。

扩展:@Autowired的用法其实有很多

  • 可以使用 @Autowired直接标注类中的字段,快速使用。

  • 可以使用 @Autowired 标注构造方法/普通的方法,这样可以方便注入多个 Bean。

  • 可以 @Autowired标注 Array (数组),Collections(集合),甚至是 Map (散列表),将所有匹配到的 Bean 注入进来。

注意:

使用 @Autowired 注入属性到 Map 中,key 必须是 String 类型。

@Autowired的多种用法

  • 8
    点赞
  • 20
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

逍遥自在”

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值