Spring使用@Autowired注解靜態(tài)實例對象方式
Spring @Autowired注解靜態(tài)實例對象
問題
最近項目小組在重新規(guī)劃工程的業(yè)務緩存,其中涉及到部分代碼重構(gòu),過程中發(fā)現(xiàn)有些工具類中的靜態(tài)方法需要依賴別的對象實例(該實例已配置在xml成Spring bean,非靜態(tài)可以用@Autowired加載正常使用),而我們知道,類加載后靜態(tài)成員是在內(nèi)存的共享區(qū),靜態(tài)方法里面的變量必然要使用靜態(tài)成員變量,這就有了如下代碼:
@Component public class TestClass { @Autowired private static AutowiredTypeComponent component; // 調(diào)用靜態(tài)組件的方法 public static void testMethod() { component.callTestMethod(); } }
編譯正常,但運行時報java.lang.NullPointerException: null異常,顯然在調(diào)用testMethod()方法時,component變量還沒被初始化,報NPE。
原因
所以,在Springframework里,我們是不能@Autowired一個靜態(tài)變量,使之成為一個Spring bean的。為什么?其實很簡單,因為當類加載器加載靜態(tài)變量時,Spring上下文尚未加載。所以類加載器不會在bean中正確注入靜態(tài)類,并且會失敗。
解決方案
方式一
將@Autowired 注解到類的構(gòu)造函數(shù)上。很好理解,Spring掃描到AutowiredTypeComponent的bean,然后賦給靜態(tài)變量component。示例如下:
@Component public class TestClass { private static AutowiredTypeComponent component; @Autowired public TestClass(AutowiredTypeComponent component) { TestClass.component = component; } // 調(diào)用靜態(tài)組件的方法 public static void testMethod() { component.callTestMethod(); } }
方式二
給靜態(tài)組件加setter方法,并在這個方法上加上@Autowired。Spring能掃描到AutowiredTypeComponent的bean,然后通過setter方法注入。示例如下:
@Component public class TestClass { private static AutowiredTypeComponent component; @Autowired public void setComponent(AutowiredTypeComponent component){ TestClass.component = component; } // 調(diào)用靜態(tài)組件的方法 public static void testMethod() { component.callTestMethod(); } }
方式三
定義一個靜態(tài)組件,定義一個非靜態(tài)組件并加上@Autowired注解,再定義一個初始化組件的方法并加上@PostConstruct注解。這個注解是JavaEE引入的,作用于servlet生命周期的注解,你只需要知道,用它注解的方法在構(gòu)造函數(shù)之后就會被調(diào)用。示例如下:
@Component public class TestClass { private static AutowiredTypeComponent component; @Autowired private AutowiredTypeComponent autowiredComponent; @PostConstruct private void beforeInit() { component = this.autowiredComponent; } // 調(diào)用靜態(tài)組件的方法 public static void testMethod() { component.callTestMethod(); } }
方式四
直接用Spring框架工具類獲取bean,定義成局部變量使用。但有弊端:如果該類中有多個靜態(tài)方法多次用到這個組件則每次都要這樣獲取,個人不推薦這種方式。示例如下:
public class TestClass { // 調(diào)用靜態(tài)組件的方法 public static void testMethod() { AutowiredTypeComponent component = SpringApplicationContextUtil.getBean("component"); component.callTestMethod(); } }
總結(jié)
在上面的代碼示例中,我每個類都加了@Component注解,其實可以根據(jù)需要進行變更,比如這個類是處理業(yè)務邏輯,可以換成@Service;這個類是處理請求進行轉(zhuǎn)發(fā)或重定向的,可以換成@Controller(是Spring-mvc的注解);這個類是專門用來操作Dao的就@Repository。
Spring的注解幫你做了一件很有意義的事:就是它們對應用進行了分層,這樣就能將請求處理、業(yè)務邏輯處理、數(shù)據(jù)庫操作處理分離出來,為代碼解耦,也方便了項目的開發(fā)和維護。
Spring容器bean加載機制用到了Java的反射,這里先不作贅述,以后會專門寫一篇文章來總結(jié)Java反射在Spring的IoC和AoP中的應用。
@Autowired注解和靜態(tài)方法
一、業(yè)務場景
spring框架應用中有些靜態(tài)方法需要依賴被容器管理的類,就像這樣:
@Component public class Test { @Autowired private static UserService userService; public static void test() { userService.test(); } }
這樣一定會報java.lang.NullPointerException: null異常。
二、原理剖析
靜態(tài)變量、類變量不是對象的屬性,而是一個類的屬性,所以靜態(tài)方法是屬于類(class)的,普通方法才是屬于實體對象(也就是New出來的對象)的,spring注入是在容器中實例化對象,所以不能使用靜態(tài)方法。
而使用靜態(tài)變量、類變量擴大了靜態(tài)方法的使用范圍。靜態(tài)方法在spring是不推薦使用的,依賴注入的主要目的,是讓容器去產(chǎn)生一個對象的實例,然后在整個生命周期中使用他們,同時也讓testing工作更加容易。
一旦你使用靜態(tài)方法,就不再需要去產(chǎn)生這個類的實例,這會讓testing變得更加困難,同時你也不能為一個給定的類,依靠注入方式去產(chǎn)生多個具有不同的依賴環(huán)境的實例,這種static field是隱含共享的,并且是一種global全局狀態(tài),spring同樣不推薦這樣去做。
三、解決方法
1、將@Autowire加到構(gòu)造方法上
@Component public class Test { private static UserService userService; @Autowired public Test(UserService userService) { Test.userService = userService; } }
2、用@PostConstruct注解
@Component public class Test { private static UserService userService; @Autowired private UserService userService2; @PostConstruct public void beforeInit() { userService = userService2; } }
以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關(guān)文章
Java操作redis實現(xiàn)增刪查改功能的方法示例
這篇文章主要介紹了Java操作redis實現(xiàn)增刪查改功能的方法,涉及java操作redis數(shù)據(jù)庫的連接、設(shè)置、增刪改查、釋放資源等相關(guān)操作技巧,需要的朋友可以參考下2017-08-08詳解Java中String,StringBuffer和StringBuilder的使用
這篇文章主要為大家詳細介紹了Java中String,StringBuffer和StringBuilder三者的區(qū)別以及使用,文中的少了講解詳細,感興趣的可以了解一下2022-07-07詳解基于Spring Data的領(lǐng)域事件發(fā)布
這篇文章主要介紹了詳解基于Spring Data的領(lǐng)域事件發(fā)布,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2020-04-04java ssm框架的controller實現(xiàn)向頁面?zhèn)鬟f參數(shù)
這篇文章主要介紹了java ssm框架的controller實現(xiàn)向頁面?zhèn)鬟f參數(shù),具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-05-05Spring?Boot?使用?SSE?方式向前端推送數(shù)據(jù)詳解
這篇文章主要介紹了Spring?Boot?使用SSE方式向前端推送數(shù)據(jù)詳解,SSE簡單的來說就是服務器主動向前端推送數(shù)據(jù)的一種技術(shù),它是單向的,也就是說前端是不能向服務器發(fā)送數(shù)據(jù)的2022-08-08Java中的StackOverflowError錯誤問題及解決方法
這篇文章主要介紹了Java中的StackOverflowError錯誤,在本文中,我們仔細研究了StackOverflower錯誤,包括Java代碼如何導致它,以及我們?nèi)绾卧\斷和修復它,需要的朋友可以參考下2022-07-07Java編程實現(xiàn)向文本文件中讀取數(shù)據(jù)之Scanner用法示例
這篇文章主要介紹了Java編程實現(xiàn)向文本文件中讀取數(shù)據(jù)之Scanner用法,結(jié)合實例形式分析了java使用Scanner類讀取文本文件相關(guān)操作技巧與注意事項,需要的朋友可以參考下2018-03-03IDEA調(diào)試技巧條件斷點實現(xiàn)步驟詳解
這篇文章主要介紹了IDEA調(diào)試技巧條件斷點實現(xiàn)步驟詳解,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下2020-09-09