亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

Springboot怎么解決循環(huán)依賴(基于單例 Bean)

 更新時間:2025年05月27日 10:36:30   作者:YY-帆S  
循環(huán)依賴指兩個或多個 Bean 之間相互依賴,導(dǎo)致在創(chuàng)建 Bean 的過程中出現(xiàn)“死循環(huán)”,增加了對象依賴的混亂性,依賴關(guān)系變得錯綜復(fù)雜,下面給大家介紹Springboot怎么解決循環(huán)依賴,感興趣的朋友一起看看吧

一、什么是循環(huán)依賴?

循環(huán)依賴(Circular Dependency) 指兩個或多個 Bean 之間相互依賴,導(dǎo)致在創(chuàng)建 Bean 的過程中出現(xiàn)“死循環(huán)”,增加了對象依賴的混亂性,依賴關(guān)系變得錯綜復(fù)雜。

常見三種類型的循環(huán)依賴:

類型舉例Spring 是否能解決
構(gòu)造器注入循環(huán)依賴A → B → A(構(gòu)造方法注入)?
Setter / 字段注入循環(huán)依賴A → B → A(@Autowired)?
Prototype 范圍循環(huán)依賴A(原型) → B(原型) → A?

1. 構(gòu)造器注入循環(huán)依賴(Spring ?無法解決)

@Component
public class A {
    private final B b;
    @Autowired
    public A(B b) {
        this.b = b;
    }
}
@Component
public class B {
    private final A a;
    @Autowired
    public B(A a) {
        this.a = a;
    }
}

? 結(jié)果:

報錯:BeanCurrentlyInCreationException: Requested bean is currently in creation: Is there an unresolvable circular reference?

原因:Spring 必須先構(gòu)造 A 才能注入 B,但 B 的構(gòu)造又依賴 A,導(dǎo)致死循環(huán),無法通過三級緩存提前暴露 Bean

2. 字段(或 setter)注入循環(huán)依賴(Spring ?能自動解決)

@Component
public class A {
    @Autowired
    private B b;
}
@Component
public class B {
    @Autowired
    private A a;
}

? 結(jié)果:

Spring 能自動解決,應(yīng)用成功啟動。

  • 原因:Spring 會先構(gòu)造出一個“空的 A 實例”,將其工廠加入三級緩存,B 注入 A 時就能拿到早期引用,從而打破循環(huán)。
  • 前提:spring.main.allow-circular-references: true,必須開啟情況下才能自動解決。然Spring 6.0 起(包括 Spring Boot 3.x)默認(rèn)為false

3. 原型作用域的循環(huán)依賴(Spring ?無法解決)

@Component
@Scope("prototype")
public class A {
    @Autowired
    private B b;
}
@Component
@Scope("prototype")
public class B {
    @Autowired
    private A a;
}

? 結(jié)果:

報錯:BeanCurrentlyInCreationException(創(chuàng)建過程中找不到可注入的 Bean)

原因:Spring 不緩存 prototype Bean 的創(chuàng)建過程,無法通過三級緩存解決依賴鏈,原型 Bean 不參與依賴管理

二、Spring 如何解決循環(huán)依賴(基于單例 Bean)

Spring 采用一種經(jīng)典的 三級緩存機(jī)制(3-level cache) 來解決循環(huán)依賴。這個機(jī)制存在于DefaultSingletonBeanRegistry中。

?? 前提:僅對 @Scope("singleton") 且使用字段或 setter 注入有效!

1. Bean 創(chuàng)建流程概覽(以 A → B → A 為例)

? Step-by-step:

創(chuàng)建 A 實例(構(gòu)造函數(shù)執(zhí)行);

A被標(biāo)記為“正在創(chuàng)建”,并將一個工廠(ObjectFactory)放入三級緩存

A 依賴 B → Spring 創(chuàng)建 B;

B 構(gòu)造完成,發(fā)現(xiàn)依賴 A → 嘗試獲取 A;

Spring 發(fā)現(xiàn) A 正在創(chuàng)建 → 從三級緩存拿到 ObjectFactory 生成早期 A 對象 → 放入二級緩存;

B 成功注入 A,初始化完成;

回到 A,完成初始化。

整個過程中 Spring 使用緩存提前暴露未完成的 A 實例,從而打破了循環(huán)。

2. 三級緩存詳解

緩存層級名稱描述作用
一級緩存singletonObjects完全初始化完成的 Bean最終返回 Bean 實例
二級緩存earlySingletonObjects早期曝光的 Bean 實例用于依賴注入
三級緩存singletonFactories創(chuàng)建早期 Bean 的工廠延遲暴露 Bean 引用,支持代理等

Spring 將 Bean 提前曝光的流程:

singletonFactories -> earlySingletonObjects -> singletonObjects

3. 核心方法說明(來自源碼)

在 Spring 源碼中,關(guān)鍵方法如下:

// DefaultSingletonBeanRegistry.java
// 一級緩存
private final Map<String, Object> singletonObjects = new ConcurrentHashMap<>();
// 二級緩存
private final Map<String, Object> earlySingletonObjects = new HashMap<>();
// 三級緩存
private final Map<String, ObjectFactory<?>> singletonFactories = new HashMap<>();
  • 在創(chuàng)建 Bean 前:Spring 把一個生成 Bean 的工廠方法放入三級緩存。
  • 在注入依賴時:發(fā)現(xiàn)依賴的是一個“正在創(chuàng)建”的 Bean,就會去三級緩存中拿工廠生產(chǎn)早期對象。
  • 最后再完成依賴注入,放入一級緩存,清除早期引用。

三、業(yè)務(wù)開發(fā)者解決循環(huán)依賴的方法

1、使用@Lazy懶加載依賴

使用`@Lazy`注解延遲注入依賴屬性。

@Component
public class A {
    @Autowired
    @Lazy
    private B b;
}

2、將依賴的代碼移入新類,打破依賴閉環(huán)。

A → MiddleService → B

3、在方法中動態(tài)調(diào)用spring容器的getBean方法獲取依賴,達(dá)到延遲獲取bean,避免類中直接注入循環(huán)依賴的bean

使用 ObjectFactory 或 ApplicationContext.getBean() 延遲獲取 Bean

@Component
@Scope("prototype")
public class A {
    @Autowired
    private ObjectFactory<B> bFactory;
    public void use() {
        B b = bFactory.getObject(); // 延遲獲取
    }
}

4、改為 setter 或字段注入(避免構(gòu)造器注入)

構(gòu)造器注入是“強(qiáng)依賴”,無法提前暴露:

@Component
public class A {
    private B b;
    @Autowired
    public void setB(B b) {
        this.b = b;
    }
}

5、使用@PostConstruct 或 工廠方法延遲注入

將依賴注入放到初始化之后:

@Component
public class A {
    private final B b;
    public A(B b) {
        this.b = b;
    }
    @PostConstruct
    public void init() {
        // 在這里安全使用 b
    }
}

6、開啟Spring Boot循環(huán)依賴(不推薦,除非必要)。

spring:
    main:
        allow-circular-references: true

到此這篇關(guān)于Springboot怎么解決循環(huán)依賴(基于單例 Bean)的文章就介紹到這了,更多相關(guān)Springboot循環(huán)依賴內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • Java利用移位運(yùn)算將int型分解成四個byte型的方法

    Java利用移位運(yùn)算將int型分解成四個byte型的方法

    今天小編就為大家分享一篇關(guān)于Java利用移位運(yùn)算將int型分解成四個byte型的方法,小編覺得內(nèi)容挺不錯的,現(xiàn)在分享給大家,具有很好的參考價值,需要的朋友一起跟隨小編來看看吧
    2018-12-12
  • Java中ArrayList類詳細(xì)介紹

    Java中ArrayList類詳細(xì)介紹

    這篇文章主要介紹了Java中ArrayList類詳細(xì)介紹的相關(guān)資料,需要的朋友可以參考下
    2017-04-04
  • java如何獲取指定文件夾下的所有文件名

    java如何獲取指定文件夾下的所有文件名

    這篇文章主要介紹了java如何獲取指定文件夾下的所有文件名問題,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教
    2024-01-01
  • Java圖形化界面編程介紹

    Java圖形化界面編程介紹

    這篇文章主要介紹了Java圖形化界面編程,形化界面編程可以直接的看到每一步操作帶來的效果,相對于傳統(tǒng)編程盯著黑框框?qū)W起來是非常非常有意思的,想了解更多的小伙伴請參考下面文章的詳細(xì)內(nèi)容
    2022-01-01
  • Java注解Annotaton詳解

    Java注解Annotaton詳解

    Java 注解(Annotation)又稱 Java 標(biāo)注,是 JDK5.0 引入的一種注釋機(jī)制,文中給大家介紹了三種基本的Annotaton,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友參考下吧
    2022-05-05
  • SpringBoot+slf4j線程池全鏈路調(diào)用日志跟蹤問題及解決思路(二)

    SpringBoot+slf4j線程池全鏈路調(diào)用日志跟蹤問題及解決思路(二)

    本文主要給大家介紹如何實現(xiàn)子線程中的traceId日志跟蹤,本文通過封裝Callable為例給大家介紹的非常詳細(xì),需要的朋友一起看看吧
    2021-05-05
  • Java字符判斷的小例子

    Java字符判斷的小例子

    從鍵盤上輸入一個字符串,遍歷該字符串中的每個字符,若該字符為小寫字母,則輸出“此字符是小寫字母”;若為大寫字母,則輸出“此字符為大寫字母”;否則輸出“此字符不是字母”
    2013-09-09
  • 解決springboot啟動報錯bean找不到的問題

    解決springboot啟動報錯bean找不到的問題

    這篇文章主要介紹了解決springboot啟動報錯bean找不到原因,本文給大家分享完美解決方案,通過圖文相結(jié)合給大家介紹的非常詳細(xì),需要的朋友可以參考下
    2023-03-03
  • Java 判斷字符串a(chǎn)和b是否互為旋轉(zhuǎn)詞

    Java 判斷字符串a(chǎn)和b是否互為旋轉(zhuǎn)詞

    本篇文章主要介紹了判斷字符串a(chǎn)和b是否互為旋轉(zhuǎn)詞的相關(guān)知識,具有很好的參考價值。下面跟著小編一起來看下吧
    2017-05-05
  • 深入Java7的一些新特性以及對腳本語言支持API的介紹

    深入Java7的一些新特性以及對腳本語言支持API的介紹

    本篇文章是對Java7的一些新特性以及對腳本語言支持API的概述,需要的朋友參考下
    2013-05-05

最新評論