今天在做項目的時候突然遇到一個問題:啟動服務器的時候spring
沒報錯,可是當我訪問某個頁面的時候spring報Request bean is currently in creation: is there an unresolvable circular reference
的錯誤,后來查了些資源終于找出來了原因,這里和大家分享一下;
首先產生這個錯誤的原因是因為spring循環注入了,什么是循環注入?舉個列子我有一個類A,A有一個構造器里面的參數是類B,然后類B里面有個構造器參數是類C,類C里面有個構造器參數是類A,就是我們會發現其實引用循環了A 里面有B的引用,B里面有C的引用,C里面又有A的引用。
循環依賴又分為構造器循環依賴和set
循環依賴:
首先講一下構造器的循環依賴:
public class A{ public A(B b) { }}
public class B{ public B(C c) { }}
public class C{ public C(A a) { }}
當我們用spring來加載A的時候spring的流程是這樣的:
1:spring創建A首先去當前創建池中去查找當前A是否在創建,如果發明沒有創建則準備其構造器需要的參數B,然后把創建A的標識放入當前創建池中。
2:spring創建B首先去當前創建池中去查找當前B是否在創建,如果發現沒有創建則準備其構造器需要的參數C,然后把創建B的標識放入當前創建池中。
3:spring創建C首先去當前創建池中去查找當前C是否在創建,如果發現沒有創建則準備其構造器需要的參數A,然后把創建C的標識放入當前創建池中。
4:spring創建C需要的A,這個時候會發現在當前創建池中已經有A的標識,A正在創建中則拋出BeanCurrentlyInCreationException。
構造器的循環注入是沒有辦法解決的,所以只能我們避免.
接下來看下set方式的循環注入:
set方式的循環注入分2種情況,第一種情況是可以解決的循環注入就是單列情況下。第二種情況就是無法解決的循環注入就是多列情況下,下面分析一下原因:
先看第一種情況,還是拿上面的ABC3個類來說明問題,只不過這次不是構造器里面的參數,而是換成他們的成員變量,然后通過set方式類注入,這里代碼就不寫了直接講下:
單列下set方式的注入流程是這樣的:
1:spring創建A,首先根據其無參構造器創建一個對象A,然后提前暴露出創建出來的這個A對象,然后再當前的創建池中放入創建A的標識,然后進行set方法注入B。
2:spring創建B,首先根據其無參構造器創建一個對象B,然后提前暴露出創建出來的這個B對象,然后在當前的創建池中放入創建B的標識,然后進行set方法的注入C。
3:spring創建C,首先根據其無參構造器創建一個對象C,然后提前暴露出創建處理的這個C對象,然后在當前的創建池中放入創建C的標識,然后進行set方法的注入A。
4:在第三步注入A的時候由于提前暴露出來了創建出來的A對象所以不會報BeanCurrentlyInCreationException的錯誤。
多列下set方式的循環注入不能解決的原因是在多列的情況下,當創建對象的時候spring不會提前暴露創建處理的對象A,這樣的話則會和構造器循環注入出現一樣的情況最終導致報錯。
解決辦法:
去掉最后一層的注入或者中間某一層的注入。通過spring getBean
的方式去獲取對象
講完了。講的不對的地方謝謝提出來。
總結
以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,謝謝大家對VeVb武林網的支持。
新聞熱點
疑難解答
圖片精選