一、业务场景
spring框架应用中有些静态方法需要依赖被容器管理的类,就像这样:
@Component public class Test {@Autowired </span><span style="color: #0000ff;">private</span> <span style="color: #0000ff;">static</span><span style="color: #000000;"> UserService userService; </span><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">static</span> <span style="color: #0000ff;">void</span><span style="color: #000000;"> test() { userService.test(); }
}
这样一定会报java.lang.NullPointerException: null异常。
二、原理剖析
静态变量、类变量不是对象的属性,而是一个类的属性,所以静态方法是属于类(class)的,普通方法才是属于实体对象(也就是New出来的对象)的,spring注入是在容器中实例化对象,所以不能使用静态方法。
而使用静态变量、类变量扩大了静态方法的使用范围。静态方法在spring是不推荐使用的,依赖注入的主要目的,是让容器去产生一个对象的实例,然后在整个生命周期中使用他们,同时也让testing工作更加容易。
一旦你使用静态方法,就不再需要去产生这个类的实例,这会让testing变得更加困难,同时你也不能为一个给定的类,依靠注入方式去产生多个具有不同的依赖环境的实例,这种static field是隐含共享的,并且是一种global全局状态,spring同样不推荐这样去做。
三、解决方法
1、将@Autowire加到构造方法上
@Component public class Test {</span><span style="color: #0000ff;">private</span> <span style="color: #0000ff;">static</span><span style="color: #000000;"> UserService userService; @Autowired </span><span style="color: #0000ff;">public</span><span style="color: #000000;"> Test(UserService userService) { Test.userService </span>=<span style="color: #000000;"> userService; } </span><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">static</span> <span style="color: #0000ff;">void</span><span style="color: #000000;"> test() { userService.test(); }
}
2、用@PostConstruct注解
@Component public class Test {</span><span style="color: #0000ff;">private</span> <span style="color: #0000ff;">static</span><span style="color: #000000;"> UserService userService; @Autowired </span><span style="color: #0000ff;">private</span><span style="color: #000000;"> UserService userService2; @PostConstruct </span><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">void</span><span style="color: #000000;"> beforeInit() { userService </span>=<span style="color: #000000;"> userService2; } </span><span style="color: #0000ff;">public</span> <span style="color: #0000ff;">static</span> <span style="color: #0000ff;">void</span><span style="color: #000000;"> test() { userService.test(); }
}