一 發生的場景
好幾次有個同事因為把static用到Spring的@Autowired上,導致注入的對象一直報空指針,他一直找不到錯誤在哪里,來問我,其實我以前也不知道這個問題,但我根據Spring容器的特點判定,他調用的對象與注入的對象不是一個對象,就告訴他:static的加載順序是在@Autowired之前;之后查資料才知道其實不是這樣。。。
二 原理剖析
靜態變量、類變量不是對象的屬性,而是一個類的屬性,所以靜態方法是屬於類(class)的,普通方法才是屬於實體對象(也就是New出來的對象)的,spring注入是在容器中實例化對象,所以不能使用靜態方法。而使用靜態變量、類變量擴大了靜態方法的使用范圍。靜態方法在spring是不推薦使用的,依賴注入的主要目的,是讓容器去產生一個對象的實例,然后在整個生命周期中使用他們,同時也讓testing工作更加容易。一旦你使用靜態方法,就不再需要去產生這個類的實例,這會讓testing變得更加困難,同時你也不能為一個給定的類,依靠注入方式去產生多個具有不同的依賴環境的實例,這種static field是隱含共享的,並且是一種global全局狀態,Spring同樣不推薦這樣去做。
三 解決方案
1、將@Autowire加到構造方法上
@Component public class Test { private static UserService userService; @Autowired public Test(UserService userService) { Test.userService = userService; } public static void test() { userService.test(); } }
2、用@PostConstruct注解
@Component public class Test { private static UserService userService; @Autowired private UserService userService2; @PostConstruct public void beforeInit() { userService = userService2; } public static void test() { userService.test(); } }
