一、業務場景
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; }