最近应该项目的需要,需要使用一个工具类来访问数据库。
但是这个工具类又被定义成静态访问了。
我们也需要设置一个静态变量来访问数据库。
@Autowired
private static VisaRepository visaRepository;
private static VisaCheckeeRepository visaCheckeeRepository;
复制代码
上面的代码在编译的时候是没有问题的。
但是在程序运行的时候提示空对象异常。
类加载后静态成员是在内存的共享区,静态方法里面的变量必然要使用静态成员变量。
通过日志我们可以非常明确的知道上面异常的主要原因就是因为 VisaRepository 这个变量没有初始化,简单来说就是没有被 @Autowired 上去。
问题和解决
在 Spring 框架中,不能 @Autowired 一个静态变量,使之成为一个 Spring bean。
这是因为当类加载器加载静态变量时,Spring 上下文尚未加载。所以类加载器不会在 bean 中正确注入静态类。
这个和静态变量这个属性有关的,因为静态变量总是先于 Spring 的 上下文加载。
使用构造函数
其实 IDEA 已经非常明确的建议我们不要使用变量 @Autowired 的方式。
而建议使用构造方法或者 Setter 的方式。
Marks a constructor, field, setter method, or config method as to be autowired by Spring's dependency injection facilities. This is an alternative to the JSR-330 javax.inject.Inject annotation, adding required-vs-optional semantics.
复制代码
在这个时候,我们只需要简单的将 @Autowired 放到构造方法上。
按照下面这样写就可以了。
private static VisaRepository visaRepository;
private static VisaCheckeeRepository visaCheckeeRepository;
@Autowired
public CheckeeUtils(VisaCheckeeRepository visaCheckeeRepository, VisaRepository visaRepository) {
this.visaCheckeeRepository = visaCheckeeRepository;
this.visaRepository = visaRepository;
}
复制代码
setter
给静态变量加一个 setter 方法,并在这个方法上加上 @Autowired。
Spring 就能扫描到 AutowiredTypeComponent 的 bean,然后通过 setter 方法注入。
@Autowired
public static void setVisaRepository(VisaRepository visaRepository) {
CheckeeUtils.visaRepository = visaRepository;
}
复制代码
定义 2 个变量
可以定义一个静态变量,一个非静态变量。
然后使用 @PostConstruct 注解。
这个注解是 JavaEE 使用的,我们通过注解就知道,这个注解就是在构造方法被执行后下一个执行的方法。
我们可以在这里对我们的静态变量初始化。
@Component
public class TestClass {
private static AutowiredTypeComponent component;
@Autowired
private AutowiredTypeComponent autowiredComponent;
@PostConstruct
private void beforeInit() {
component = this.autowiredComponent;
}
// 调用静态组件的方法
public static void testMethod() {
component.callTestMethod();
}
}
复制代码
使用 Spring 的工具类获取 Bean
这个方法就是直接调用 Spring 的上下文工具来获得组件。
AutowiredTypeComponent component = SpringApplicationContextUtil.getBean("component");
component.callTestMethod();
复制代码
这个方法实在太麻烦,每一次要用的时候都要调用一次,增加不少容易的代码。
总结
这个问题就是 Spring 的 Bean 在什么时候初始化的问题。
如果没有初始化的话,是没有办法直接调用和自动加载的。
根据官方的提示,不要使用变量上的自动加载,使用构造方法的自动加载就可以了,这个也是官方推荐的方式。
https://www.ossez.com/t/spring-autowired/14074
评论