Spring解決迴圈依賴,你真的懂了嗎?

来源:https://www.cnblogs.com/Chenjiabing/archive/2020/03/18/12521178.html
-Advertisement-
Play Games

導讀 前幾天發表的文章 "SpringBoot多數據源動態切換" 和 "SpringBoot整合多數據源的巨坑" 中,提到了一個坑就是動態數據源添加@Primary介面就會造成迴圈依賴異常,如下圖: 這個就是典型的構造器依賴,詳情請看上面兩篇文章,這裡不再詳細贅述了。本篇文章將會從源碼深入解析Spr ...


導讀

微信所有碼猿技術專欄

  • 這個就是典型的構造器依賴,詳情請看上面兩篇文章,這裡不再詳細贅述了。本篇文章將會從源碼深入解析Spring是如何解決迴圈依賴的?為什麼不能解決構造器的迴圈依賴?

什麼是迴圈依賴

  • 簡單的說就是A依賴B,B依賴C,C依賴A這樣就構成了迴圈依賴。

微信搜索碼猿技術專欄

  • 迴圈依賴分為構造器依賴和屬性依賴,眾所周知的是Spring能夠解決屬性的迴圈依賴(set註入)。下文將從源碼角度分析Spring是如何解決屬性的迴圈依賴。

思路

  • 如何解決迴圈依賴,Spring主要的思路就是依據三級緩存,在實例化A時調用doGetBean,發現A依賴的B的實例,此時調用doGetBean去實例B,實例化的B的時候發現又依賴A,如果不解決這個迴圈依賴的話此時的doGetBean將會無限迴圈下去,導致記憶體溢出,程式奔潰。spring引用了一個早期對象,並且把這個"早期引用"並將其註入到容器中,讓B先完成實例化,此時A就獲取B的引用,完成實例化。

三級緩存

  • Spring能夠輕鬆的解決屬性的迴圈依賴正式用到了三級緩存,在AbstractBeanFactory中有詳細的註釋。
    /**一級緩存,用於存放完全初始化好的 bean,從該緩存中取出的 bean 可以直接使用*/
    private final Map<String, Object> singletonObjects = new ConcurrentHashMap<>(256);

    /**三級緩存 存放 bean 工廠對象,用於解決迴圈依賴*/
    private final Map<String, ObjectFactory<?>> singletonFactories = new HashMap<>(16);

    /**二級緩存 存放原始的 bean 對象(尚未填充屬性),用於解決迴圈依賴*/
    private final Map<String, Object> earlySingletonObjects = new HashMap<>(16);
  • 一級緩存:singletonObjects,存放完全實例化屬性賦值完成的Bean,直接可以使用。
  • 二級緩存:earlySingletonObjects,存放早期Bean的引用,尚未屬性裝配的Bean
  • 三級緩存:singletonFactories,三級緩存,存放實例化完成的Bean工廠。

開擼

  • 先上一張流程圖看看Spring是如何解決迴圈依賴的

微信搜索碼猿技術專欄

  • 上圖標記藍色的部分都是涉及到三級緩存的操作,下麵我們一個一個方法解析

【1】 getSingleton(beanName):源碼如下:

        //查詢緩存
        Object sharedInstance = getSingleton(beanName);
        //緩存中存在並且args是null
        if (sharedInstance != null && args == null) {
            //.......省略部分代碼
            
            //直接獲取Bean實例
            bean = getObjectForBeanInstance(sharedInstance, name, beanName, null);
        }
        
    //getSingleton源碼,DefaultSingletonBeanRegistry#getSingleton
protected Object getSingleton(String beanName, boolean allowEarlyReference) {
        //先從一級緩存中獲取已經實例化屬性賦值完成的Bean
        Object singletonObject = this.singletonObjects.get(beanName);
        //一級緩存不存在,並且Bean正處於創建的過程中
        if (singletonObject == null && isSingletonCurrentlyInCreation(beanName)) {
            synchronized (this.singletonObjects) {
                //從二級緩存中查詢,獲取Bean的早期引用,實例化完成但是未賦值完成的Bean
                singletonObject = this.earlySingletonObjects.get(beanName);
                //二級緩存中不存在,並且允許創建早期引用(二級緩存中添加)
                if (singletonObject == null && allowEarlyReference) {
                    //從三級緩存中查詢,實例化完成,屬性未裝配完成
                    ObjectFactory<?> singletonFactory = this.singletonFactories.get(beanName);
                    if (singletonFactory != null) {
                        singletonObject = singletonFactory.getObject();
                            //二級緩存中添加
                        this.earlySingletonObjects.put(beanName, singletonObject);
                        //從三級緩存中移除
                        this.singletonFactories.remove(beanName);
                    }
                }
            }
        }
        return singletonObject;
    }
        

  • 從源碼可以得知,doGetBean最初是查詢緩存,一二三級緩存全部查詢,如果三級緩存存在則將Bean早期引用存放在二級緩存中並移除三級緩存。(升級為二級緩存)

【2】addSingletonFactory:源碼如下


        //中間省略部分代碼。。。。。
        //創建Bean的源碼,在AbstractAutowireCapableBeanFactory#doCreateBean方法中
        if (instanceWrapper == null) {
            //實例化Bean
            instanceWrapper = createBeanInstance(beanName, mbd, args);
        }
        //允許提前暴露
        if (earlySingletonExposure) {
            //添加到三級緩存中
            addSingletonFactory(beanName, () -> getEarlyBeanReference(beanName, mbd, bean));
        }
        try {
            //屬性裝配,屬性賦值的時候,如果有發現屬性引用了另外一個Bean,則調用getBean方法
            populateBean(beanName, mbd, instanceWrapper);
            //初始化Bean,調用init-method,afterproperties方法等操作
            exposedObject = initializeBean(beanName, exposedObject, mbd);
        }
        }

//添加到三級緩存的源碼,在DefaultSingletonBeanRegistry#addSingletonFactory
protected void addSingletonFactory(String beanName, ObjectFactory<?> singletonFactory) {
        synchronized (this.singletonObjects) {
            //一級緩存中不存在
            if (!this.singletonObjects.containsKey(beanName)) {
                //放入三級緩存
                this.singletonFactories.put(beanName, singletonFactory);
                //從二級緩存中移除,
                this.earlySingletonObjects.remove(beanName);
                this.registeredSingletons.add(beanName);
            }
        }
    }
  • 從源碼得知,Bean在實例化完成之後會直接將未裝配的Bean工廠存放在三級緩存中,並且移除二級緩存

【3】addSingleton:源碼如下:

//獲取單例對象的方法,DefaultSingletonBeanRegistry#getSingleton
//調用createBean實例化Bean
singletonObject = singletonFactory.getObject();

//。。。。中間省略部分代碼  

//doCreateBean之後才調用,實例化,屬性賦值完成的Bean裝入一級緩存,可以直接使用的Bean
addSingleton(beanName, singletonObject);

//addSingleton源碼,在DefaultSingletonBeanRegistry#addSingleton方法中
protected void addSingleton(String beanName, Object singletonObject) {
        synchronized (this.singletonObjects) {
            //一級緩存中添加
            this.singletonObjects.put(beanName, singletonObject);
            //移除三級緩存
            this.singletonFactories.remove(beanName);
            //移除二級緩存
            this.earlySingletonObjects.remove(beanName);
            this.registeredSingletons.add(beanName);
        }
    }


                
  • 總之一句話,Bean添加到一級緩存,移除二三級緩存。

擴展

【1】為什麼Spring不能解決構造器的迴圈依賴?

  • 從流程圖應該不難看出來,在Bean調用構造器實例化之前,一二三級緩存並沒有Bean的任何相關信息,在實例化之後才放入三級緩存中,因此當getBean的時候緩存並沒有命中,這樣就拋出了迴圈依賴的異常了。

【2】為什麼多實例Bean不能解決迴圈依賴?

  • 多實例Bean是每次創建都會調用doGetBean方法,根本沒有使用一二三級緩存,肯定不能解決迴圈依賴。

總結

  • 根據以上的分析,大概清楚了Spring是如何解決迴圈依賴的。假設A依賴B,B依賴A(註意:這裡是set屬性依賴)分以下步驟執行:
  1. A依次執行doGetBean、查詢緩存、createBean創建實例,實例化完成放入三級緩存singletonFactories中,接著執行populateBean方法裝配屬性,但是發現有一個屬性是B的對象。
  2. 因此再次調用doGetBean方法創建B的實例,依次執行doGetBean、查詢緩存、createBean創建實例,實例化完成之後放入三級緩存singletonFactories中,執行populateBean裝配屬性,但是此時發現有一個屬性是A對象。
  3. 因此再次調用doGetBean創建A的實例,但是執行到getSingleton查詢緩存的時候,從三級緩存中查詢到了A的實例(早期引用,未完成屬性裝配),此時直接返回A,不用執行後續的流程創建A了,那麼B就完成了屬性裝配,此時是一個完整的對象放入到一級緩存singletonObjects中。
  4. B創建完成了,則A自然完成了屬性裝配,也創建完成放入了一級緩存singletonObjects中。
  • Spring三級緩存的應用完美的解決了迴圈依賴的問題,下麵是迴圈依賴的解決流程圖。

微信搜索碼猿技術專欄

  • 如果覺得作者寫的好,有所收穫的話,點個關註推薦一下喲!!!
    微信搜索公眾號碼猿技術專欄

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • vue前端項目中開發基於甘特圖的項目計劃模塊 參考鏈接 相比以前jquery的資料,vue的甘特圖插件少很多,中文資料更是少的可憐,以下兩個鏈接是在網上搜到相對還不錯的甘特圖插件 "https://www.cnblogs.com/liang715200/p/12029640.html" "https ...
  • [toc] 運用領域模型 交流與語言的使用 非原創,感謝《領域驅動設計》這本書 領域模型可成為軟體項目通用語言的核心。該模型是一組得自於項目人員頭腦中的概念,以及反映了領域深層含義的術語和關係。這些術語和相互關係提供了模型語言的語義,雖然語言是為領域量身定製的,但就技術開發而言,其依然足夠精確。正是 ...
  • [toc] 運用領域模型 消化知識 非原創,感謝《領域驅動設計》這本書 有效建模的要素 (1) 模型和實現的綁定。最初的原型雖然簡陋,但它在模型與實現之間建立了早期鏈接,而且在所有後續的迭代中我們一直在維護該鏈接。 (2) 建立了一種基於模型的語言。隨著項目的進展,雙方都能夠直接使用模型中的術語,並 ...
  • 設計模式中的原則和法則: 1、開閉原則: 開閉原則(Open Closed Principle,OCP)由勃蘭特·梅耶(Bertrand Meyer)提出,他在 1988 年的著作《面向對象軟體構造》(Object Oriented Software Construction)中提出:軟體實體應當對 ...
  • 1、最簡單的用戶系統 一個最簡單的用戶系統,只需要有用戶和身份驗證兩個模塊就夠了。如圖: 這裡提示一下:上層數據依賴下層數據。舉個慄子,就是身份驗證需要依賴用戶數據。 2、具有許可權管理的用戶系統 如果需要進行許可權管理的話,那麼就加上資源和角色模塊。同時,在身份認證之後需要按需進行鑒權。 資源和用戶都 ...
  • 本次過程僅供學習參考,請遵守相關法律法規。 首先我們分析網站:https://www.mzitu.com/all/ 不難發現,這個頁面上包含了大量的圖片鏈接,可以說是特別方便我們爬取圖片的,這是件好事。那麼我們繼續分析 這是第一頁的地址 這是第二頁的,所以我們爬取的時候只需要在鏈接後面增加“/num ...
  • OpenCV中的HAL方法調用流程分析 在OpenCV中有一些所謂HAL(Hardware Acceleration Layer)實現,看名字好像和硬體相關,其實也不盡然,可以理解為比常規的OCV實現更快的版本就好了。此文要做的就是要找到其實現或者切入流程,打通整個函數調用邏輯。本文將以 和`Gau ...
  • 一、Buffered 位元組方式 BufferedInputStream BufferedOutputStream 字元方式 BufferedReader BufferedWriter package com.bjpowernode.java_learning; import java.io.*; p ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...