一、业务场景
spring框架应用中有些静态方法需要依赖被容器管理的类,就像这样:
@Component public class Test { @Autowired private static UserService userService; public static void test() { userService.test(); } }
这样一定会报java.lang.NullPointerException: null异常。
二、原理剖析
静态变量、类变量不是对象的属性,而是一个类的属性,所以静态方法是属于类(class)的,普通方法才是属于实体对象(也就是New出来的对象)的,spring注入是在容器中实例化对象,所以不能使用静态方法。
而使用静态变量、类变量扩大了静态方法的使用范围。静态方法在spring是不推荐使用的,依赖注入的主要目的,是让容器去产生一个对象的实例,然后在整个生命周期中使用他们,同时也让testing工作更加容易。
一旦你使用静态方法,就不再需要去产生这个类的实例,这会让testing变得更加困难,同时你也不能为一个给定的类,依靠注入方式去产生多个具有不同的依赖环境的实例,这种static field是隐含共享的,并且是一种global全局状态,spring同样不推荐这样去做。
三、解决方法
网上很多的解决方法,但是都不对,
后来同事教了个方法:使用SpringUtil.getBean(类名.class)就可以了;
private static MessageRepository messageRepository; static { if(messageRepository==null){ messageRepository=SpringUtil.getBean(MessageRepository.class); } }
使用的位置:
String message = messageRepository.getOriginalMsg(cityRequest.getTraceId()); if (StringUtils.isBlank(message)) { return; }