spring静态变量如何注入

spring静态变量如何注入

背景

这种写法,是注入不了的,结果打印 null 值

@SpringBootTest
@RunWith(SpringRunner.class)
public class MyTest {
    @Autowired
    private static ApplicationContext applicationContext;

    @Test
    public void test() {
        System.out.println("初始化了吗? " + applicationContext);
    }
}

方法

方式一(推荐)

通过非静态的setter上标注@Autowired 进行注入

@SpringBootTest
@RunWith(SpringRunner.class)
public class MyTest {
    private static ApplicationContext applicationContext;
    
    @Autowired
    public void setApplicationContext(ApplicationContext applicationContext) {
        MyTest.applicationContext = applicationContext;
    }

    @Test
    public void test() {
        System.out.println("初始化了吗? " + applicationContext);
    }
}
方式二

通过 @PostConstruct 在bean构建后赋值给静态成员变量

@SpringBootTest
@RunWith(SpringRunner.class)
public class MyTest {
    private static ApplicationContext applicationContext;

    @Autowired
    private ApplicationContext applicationContextNonStatic;

    @PostConstruct
    public void post() {
        MyTest.applicationContext = applicationContextNonStatic;
    }

    @Test
    public void test() {
        System.out.println("初始化了吗? " + applicationContext);
    }
}

补充(可以不看)

思考1:哪些常见需要静态注入?

为什么我们有时候需要注入到静态变量呢? 因为我们的工具类,一般喜欢使用静态的方法,而只有静态的变量才能在静态方法里使用。比如:我有个Util是获取数据库sequence序列号的,获取序列号的方法是静态的,因此要求mapper类也是静态的。

// 【方法1】注入静态变量,并调用
@Component
public class SequenceUtil {
  private static SequenceMapper sequenceMapper;
  
  @Autowired
  public void setSequenceMapper(SequenceMapper sequenceMapper) {
    SequenceUtil.sequenceMapper = sequenceMapper;
  }
  
  public static Long getNextVal() {
        return sequenceMapper.getNextVal();
    }
}


// 【方法2】通过 SpringContextUtil 获取spring所管理的bean
public class SequenceUtil {
  public static Long getNextVal() {
        SequenceMapper sequenceMapper = SpringContextUtil.getBean(SequenceMapper.class);// SpringContextUtil.getBean("sequenceMapper");
        return sequenceMapper.getNextVal();
    }
}
思考2:为什么能通过上面的方法静态注入

其实上面的方式(如在非静态setter里进行赋值)为什么可以成功? 成功是肯定可以成功的,因为静态变量本身就可以赋值,只是其赋值的时机往后延到某个对象的setter时才被赋值而已。

所以,如果这些静态方法 getNextVal() 如果在spring容器起来之后就能正常访问这些mapper,因为spring容器起来后代表了所有的bean已经初始化好了并且通过非静态的setter赋值给静态变量了,所以肯定是非null可以使用的了。(当然如果在此之前的时机调用 getNextVal() 的话mapper还是null的。例如下面的例子:


// init 方法是在spring容器初始化好之前被调用的,因此打印出null。
// init执行后,执行setApplicationContext方法,此时静态变量才被赋值,再之后执行test()的测试方法,此时静态变量赋值了
@SpringBootTest
@RunWith(SpringRunner.class)
public class MyTest {
    @Autowired
    private static ApplicationContext applicationContext;


    @BeforeClass
    public static void init() {
        System.out.println("before=====================" + applicationContext);
    }

    @Autowired
    public void setApplicationContext(ApplicationContext applicationContext) {
        MyTest.applicationContext = applicationContext;
    }

    @Test
    public void test() {
        System.out.println("spring test ok" + applicationContext);
    }
}

其实可以想象得到,如果 MyTest 类不是单例,则这个静态变量(applicationContext)肯定是被赋值超过一次的,每个bean对象的时候都会执行setApplicationContext进行赋值

### 注入静态属性的方法 在Spring框架中,直接向静态字段注入值并非推荐的做法,因为这违反了依赖注入的原则并可能导致不可预测的行为。然而,在某些特殊情况下确实有需求这样做。 一种常见的方法是使用`@PostConstruct`注解配合自定义逻辑来设置静态变量的值: ```java import javax.annotation.PostConstruct; import org.springframework.beans.factory.annotation.Value; import org.springframework.stereotype.Component; @Component public class StaticPropertyHolder { private static String myStaticField; @Value("${my.property.value}") private String propertyValue; @PostConstruct public void init() { myStaticField = propertyValue; } public static String getMyStaticField() { return myStaticField; } } ``` 另一种方式是在配置类里通过`static setter`来进行赋值操作: ```java @Configuration public class AppConfig { @Bean public static PropertyConfigurer propertyConfigurer(@Value("${example.static}") final String value) { return new PropertyConfigurer(value); } public static class PropertyConfigurer implements BeanFactoryPostProcessor { private final String value; public PropertyConfigurer(String value){ this.value=value; MyStaticClass.setValue(this.value); // 调用静态setter } ... } } ``` 需要注意的是,上述做法虽然可行,但在实际项目开发过程中应当谨慎考虑是否真的有必要这么做。通常来说,应该尽量避免使用全局状态或共享资源,转而采用更安全的设计模式如单例模式或者工厂模式[^1]。 对于数据访问层面而言,当涉及到数据库交互时,建议利用Spring所提供的抽象层以及自动装配功能,而不是手动管理DataSource实例或其他底层细节[^2]。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值