今天在做項目的時候突然遇到一個問題:啟動服務器的時候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
循環依賴:
首先講一下構造器的循環依賴:
1
2
3
4
5
6
|
public class a { public a(b b) { } } |
1
2
3
4
5
6
|
public class b { public b(c c) { } } |
1
2
3
4
5
6
|
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
的方式去獲取對象
講完了。講的不對的地方謝謝提出來。
總結
以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,謝謝大家對服務器之家的支持。如果你想了解更多相關內容請查看下面相關鏈接
原文鏈接:https://blog.csdn.net/chengxu2011/article/details/8478290