色哟哟视频在线观看-色哟哟视频在线-色哟哟欧美15最新在线-色哟哟免费在线观看-国产l精品国产亚洲区在线观看-国产l精品国产亚洲区久久

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

什么是循環依賴?

倩倩 ? 來源:樓仔 ? 作者:樓仔 ? 2022-09-08 10:49 ? 次閱讀


7d76c324-2f1f-11ed-ba43-dac502259ad0.png

1. 基礎知識

1.1 什么是循環依賴 ?

一個或多個對象之間存在直接或間接的依賴關系,這種依賴關系構成一個環形調用,有下面 3 種方式。

7d9ea39e-2f1f-11ed-ba43-dac502259ad0.png

我們看一個簡單的 Demo,對標“情況 2”。

@Service
publicclassLouzai1{

@Autowired
privateLouzai2louzai2;

publicvoidtest1(){
}
}

@Service
publicclassLouzai2{
@Autowired
privateLouzai1louzai1;

publicvoidtest2(){
}
}

這是一個經典的循環依賴,它能正常運行,后面我們會通過源碼的角度,解讀整體的執行流程。

1.2 三級緩存

解讀源碼流程之前,spring 內部的三級緩存邏輯必須了解,要不然后面看代碼會蒙圈。

  • 第一級緩存 :singletonObjects,用于保存實例化、注入、初始化完成的 bean 實例;
  • 第二級緩存 :earlySingletonObjects,用于保存實例化完成的 bean 實例;
  • 第三級緩存 :singletonFactories,用于保存 bean 創建工廠,以便后面有機會創建代理對象。

這是最核心,我們直接上源碼:

7db71186-2f1f-11ed-ba43-dac502259ad0.png

執行邏輯:

  • 先從“第一級緩存”找對象,有就返回,沒有就找“二級緩存”;
  • 找“二級緩存”,有就返回,沒有就找“三級緩存”;
  • 找“三級緩存”,找到了,就獲取對象,放到“二級緩存”,從“三級緩存”移除。

1.3 原理執行流程

我把“情況 2”執行的流程分解為下面 3 步,是不是和“套娃”很像 ?

7de0ad48-2f1f-11ed-ba43-dac502259ad0.png

整個執行邏輯如下:

  1. 在第一層中,先去獲取 A 的 Bean,發現沒有就準備去創建一個,然后將 A 的代理工廠放入“三級緩存”(這個 A 其實是一個半成品,還沒有對里面的屬性進行注入 ),但是 A 依賴 B 的創建,就必須先去創建 B;
  2. 在第二層中,準備創建 B,發現 B 又依賴 A,需要先去創建 A;
  3. 在第三層中,去創建 A,因為第一層已經創建了 A 的代理工廠,直接從“三級緩存”中拿到 A 的代理工廠,獲取 A 的代理對象,放入“二級緩存” ,并清除“三級緩存”;
  4. 回到第二層,現在有了 A 的代理對象,對 A 的依賴完美解決(這里的 A 仍然是個半成品 ),B 初始化成功;
  5. 回到第一層,現在 B 初始化成功,完成 A 對象的屬性注入,然后再填充 A 的其它屬性,以及 A 的其它步驟(包括 AOP),完成對 A 完整的初始化功能(這里的 A 才是完整的 Bean )。
  6. 將 A 放入“一級緩存”。

為什么要用 3 級緩存 ?我們先看源碼執行流程,后面我會給出答案。

基于 Spring Boot + MyBatis Plus + Vue & Element 實現的后臺管理系統 + 用戶小程序,支持 RBAC 動態權限、多租戶、數據權限、工作流、三方登錄、支付、短信、商城等功能

  • 項目地址:https://gitee.com/zhijiantianya/ruoyi-vue-pro
  • 視頻教程:https://doc.iocoder.cn/video/

2. 源碼解讀

注意:Spring 的版本是 5.2.15.RELEASE ,否則和我的代碼不一樣!!!

上面的知識,網上其實都有,下面才是我們的重頭戲,讓你跟著樓仔,走一遍代碼流程。

2.1 代碼入口

7e0d204e-2f1f-11ed-ba43-dac502259ad0.png7e326a16-2f1f-11ed-ba43-dac502259ad0.png

這里需要多跑幾次,把前面的 beanName 跳過去,只看 louzai1。

7e50616a-2f1f-11ed-ba43-dac502259ad0.png7e72d556-2f1f-11ed-ba43-dac502259ad0.png

2.2 第一層

7e905a36-2f1f-11ed-ba43-dac502259ad0.png

進入 doGetBean(),從 getSingleton() 沒有找到對象,進入創建 Bean 的邏輯。

7eb0d770-2f1f-11ed-ba43-dac502259ad0.png7ecdbbb0-2f1f-11ed-ba43-dac502259ad0.png

進入 doCreateBean() 后,調用 addSingletonFactory()。

7ef65eb2-2f1f-11ed-ba43-dac502259ad0.png

往三級緩存 singletonFactories 塞入 louzai1 的工廠對象。

7f17fe3c-2f1f-11ed-ba43-dac502259ad0.png7f52c4a4-2f1f-11ed-ba43-dac502259ad0.png

進入到 populateBean(),執行 postProcessProperties(),這里是一個策略模式,找到下圖的策略對象。

7f71d3a8-2f1f-11ed-ba43-dac502259ad0.png

正式進入該策略對應的方法。

7f9b11b4-2f1f-11ed-ba43-dac502259ad0.png

下面都是為了獲取 louzai1 的成員對象,然后進行注入。

7fa65650-2f1f-11ed-ba43-dac502259ad0.png7fd14914-2f1f-11ed-ba43-dac502259ad0.png7fed7fd0-2f1f-11ed-ba43-dac502259ad0.png7ff82c0a-2f1f-11ed-ba43-dac502259ad0.png

進入 doResolveDependency(),找到 louzai1 依賴的對象名 louzai2

8008ff76-2f1f-11ed-ba43-dac502259ad0.png

需要獲取 louzai2 的 bean,是 AbstractBeanFactory 的方法。

8018f020-2f1f-11ed-ba43-dac502259ad0.png

正式獲取 louzai2 的 bean。

8034f6b2-2f1f-11ed-ba43-dac502259ad0.png

到這里,第一層套娃基本結束,因為 louzai1 依賴 louzai2,下面我們進入第二層套娃。

2.3 第二層

804c28dc-2f1f-11ed-ba43-dac502259ad0.png

獲取 louzai2 的 bean,從 doGetBean(),到 doResolveDependency(),和第一層的邏輯完全一樣,找到 louzai2 依賴的對象名 louzai1。

前面的流程全部省略,直接到 doResolveDependency()。

806a87fa-2f1f-11ed-ba43-dac502259ad0.png

正式獲取 louzai1 的 bean。

80776cea-2f1f-11ed-ba43-dac502259ad0.png

到這里,第二層套娃結束,因為 louzai2 依賴 louzai1,所以我們進入第三層套娃。

2.4 第三層

809c6b8a-2f1f-11ed-ba43-dac502259ad0.png

獲取 louzai1 的 bean,在第一層和第二層中,我們每次都會從 getSingleton() 獲取對象,但是由于之前沒有初始化 louzai1 和 louzai2 的三級緩存,所以獲取對象為空。

80c2d7c0-2f1f-11ed-ba43-dac502259ad0.png80e2b93c-2f1f-11ed-ba43-dac502259ad0.png

敲重點!敲重點!!敲重點!!!

到了第三層,由于第三級緩存有 louzai1 數據,這里使用三級緩存中的工廠,為 louzai1 創建一個代理對象,塞入二級緩存。

80fca392-2f1f-11ed-ba43-dac502259ad0.png

這里就拿到了 louzai1 的代理對象,解決了 louzai2 的依賴關系,返回到第二層。

2.5 返回第二層

返回第二層后,louzai2 初始化結束,這里就結束了么?二級緩存的數據,啥時候會給到一級呢?

甭著急,看這里,還記得在 doGetBean() 中,我們會通過 createBean() 創建一個 louzai2 的 bean,當 louzai2 的 bean 創建成功后,我們會執行 getSingleton(),它會對 louzai2 的結果進行處理。

812f996e-2f1f-11ed-ba43-dac502259ad0.png

我們進入 getSingleton(),會看到下面這個方法。

813aad22-2f1f-11ed-ba43-dac502259ad0.png

這里就是處理 louzai2 的 一、二級緩存的邏輯,將二級緩存清除,放入一級緩存。

815da99e-2f1f-11ed-ba43-dac502259ad0.png

2.6 返回第一層

同 2.5,louzai1 初始化完畢后,會把 louzai1 的二級緩存清除,將對象放入一級緩存。

818a0aac-2f1f-11ed-ba43-dac502259ad0.png

到這里,所有的流程結束,我們返回 louzai1 對象。

基于 Spring Cloud Alibaba + Gateway + Nacos + RocketMQ + Vue & Element 實現的后臺管理系統 + 用戶小程序,支持 RBAC 動態權限、多租戶、數據權限、工作流、三方登錄、支付、短信、商城等功能

  • 項目地址:https://gitee.com/zhijiantianya/yudao-cloud
  • 視頻教程:https://doc.iocoder.cn/video/

3. 原理深度解讀

3.1 什么要有 3 級緩存 ?

這是一道非常經典的面試題,前面已經告訴大家詳細的執行流程,包括源碼解讀,但是沒有告訴大家為什么要用 3 級緩存?

這里是重點!敲黑板!!!

我們先說“一級緩存”的作用,變量命名為 singletonObjects,結構是 Map,它就是一個單例池,將初始化好的對象放到里面,給其它線程使用,如果沒有第一級緩存,程序不能保證 Spring 的單例屬性。

“二級緩存”先放放,我們直接看“三級緩存”的作用,變量命名為 singletonFactories,結構是 Map>,Map 的 Value 是一個對象的代理工廠,所以“三級緩存”的作用,其實就是用來存放對象的代理工廠。

那這個對象的代理工廠有什么作用呢,我先給出答案,它的主要作用是存放半成品的單例 Bean,目的是為了“打破循環” ,可能大家還是不太懂,這里我再稍微解釋一下。

我們回到文章開頭的例子,創建 A 對象時,會把實例化的 A 對象存入“三級緩存”,這個 A 其實是個半成品,因為沒有完成 A 的依賴屬性 B 的注入,所以后面當初始化 B 時,B 又要去找 A,這時就需要從“三級緩存”中拿到這個半成品的 A(這里描述,其實也不完全準確,因為不是直接拿,為了讓大家好理解,我就先這樣描述),打破循環。

那我再問一個問題,為什么“三級緩存”不直接存半成品的 A,而是要存一個代理工廠呢 ?答案是因為 AOP。

在解釋這個問題前,我們看一下這個代理工廠的源碼,讓大家有一個更清晰的認識。

直接找到創建 A 對象時,把實例化的 A 對象存入“三級緩存”的代碼,直接用前面的兩幅截圖。

7ef65eb2-2f1f-11ed-ba43-dac502259ad0.png7f17fe3c-2f1f-11ed-ba43-dac502259ad0.png

下面我們主要看這個對象工廠是如何得到的,進入 getEarlyBeanReference() 方法。

81e4a674-2f1f-11ed-ba43-dac502259ad0.png8200c28c-2f1f-11ed-ba43-dac502259ad0.png8212d3fa-2f1f-11ed-ba43-dac502259ad0.png822e74d4-2f1f-11ed-ba43-dac502259ad0.png

最后一幅圖太重要了,我們知道這個對象工廠的作用:

  • 如果 A 有 AOP,就創建一個代理對象;
  • 如果 A 沒有 AOP,就返回原對象。

那“二級緩存”的作用就清楚了,就是用來存放對象工廠生成的對象,這個對象可能是原對象,也可能是個代理對象。

我再問一個問題,為什么要這樣設計呢?把二級緩存干掉不行么 ?我們繼續往下看。

3.2 能干掉第 2 級緩存么 ?

@Service
publicclassA{

@Autowired
privateBb;

@Autowired
privateCc;

publicvoidtest1(){
}
}

@Service
publicclassB{
@Autowired
privateAa;

publicvoidtest2(){
}
}

@Service
publicclassC{

@Autowired
privateAa;

publicvoidtest3(){
}
}

根據上面的套娃邏輯,A 需要找 B 和 C,但是 B 需要找 A,C 也需要找 A。

假如 A 需要進行 AOP ,因為代理對象每次都是生成不同的對象,如果干掉第二級緩存,只有第一、三級緩存:

  • B 找到 A 時,直接通過三級緩存的工廠的代理對象,生成對象 A1。
  • C 找到 A 時,直接通過三級緩存的工廠的代理對象,生成對象 A2。

看到問題沒?你通過 A 的工廠的代理對象,生成了兩個不同的對象 A1 和 A2 ,所以為了避免這種問題的出現,我們搞個二級緩存,把 A1 存下來,下次再獲取時,直接從二級緩存獲取,無需再生成新的代理對象。

所以“二級緩存”的目的是為了避免因為 AOP 創建多個對象,其中存儲的是半成品的 AOP 的單例 bean。

如果沒有 AOP 的話,我們其實只要 1、3 級緩存,就可以滿足要求。

4. 寫在最后

我們再回顧一下 3 級緩存的作用:

  • 一級緩存:為“Spring 的單例屬性”而生 ,就是個單例池,用來存放已經初始化完成的單例 Bean;
  • 二級緩存:為“解決 AOP”而生 ,存放的是半成品的 AOP 的單例 Bean;
  • 三級緩存:為“打破循環”而生 ,存放的是生成半成品單例 Bean 的工廠方法。

如果你能理解上面我說的三條,恭喜你,你對 Spring 的循環依賴理解得非常透徹!

關于循環依賴的知識,其實還有,因為篇幅原因,我就不再寫了,這篇文章的重點,一方面是告訴大家循環依賴的核心原理,另一方面是讓大家自己去 debug 代碼 ,跑跑流程,挺有意思的。

可能有同學會問 “樓哥,你之前是不是經常看源碼,然后這個流程,你是不是 debug 了很久?”

我之前其實沒怎么看過開源代碼,這個流程,前期理論知識看了 2.5 個小時,然后 debug 4.5 小時,就基本全部走通了,最難的地方,就是三層套娃,稍微有些繞。

這里也簡單說一下我看源碼的心得:

  1. 需要掌握基本的設計模式;
  2. 看源碼前,最好能找一些理論知識先看看;
  3. 學會讀英文注釋,不會的話就百度翻譯;
  4. debug 時,要克制自己,不要陷入無用的細節 ,這個最重要。

其中最難的是第 4 步,因為很多同學看 Spring 源碼,每看一個方法,就想多研究研究,這樣很容易被繞進去了,這個要學會克制,有大局觀,并能分辨哪里是核心邏輯 ,至于如何分辨,可以在網上先找些資料,如果沒有的話,就只能多看代碼了。

今天的源碼解析就到這了~


審核編輯 :李倩


聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • 源碼
    +關注

    關注

    8

    文章

    652

    瀏覽量

    29357
  • spring
    +關注

    關注

    0

    文章

    340

    瀏覽量

    14368

原文標題:痛快!SpringBoot終于幫我們禁止了Spring循環依賴!

文章出處:【微信號:芋道源碼,微信公眾號:芋道源碼】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦

    Simcenter Micred Power Tester功率循環測試儀

    SimcenterMicredPowerTester功率循環測試儀使用結合了有效功率循環和熱結構退化監測的測試硬件,評估功率半導體的熱可靠性和使用壽命。為什么選擇
    的頭像 發表于 01-09 14:33 ?111次閱讀
    Simcenter Micred Power Tester功率<b class='flag-5'>循環</b>測試儀

    深入理解C語言:循環語句的應用與優化技巧

    在程序設計中,我們常常需要重復執行某一段代碼。為了提高效率和簡化代碼,循環語句應運而生。C語言作為一門經典的編程語言,提供了多種循環控制結構,幫助程序員高效地實現重復操作。掌握循環語句的使用,不僅
    的頭像 發表于 12-07 01:11 ?231次閱讀
    深入理解C語言:<b class='flag-5'>循環</b>語句的應用與優化技巧

    掃描速率和濃度對循環伏安圖有什么影響

    循環伏安法(Cyclic Voltammetry,CV)是一種電化學測試技術,它通過在工作電極上施加一個時間依賴的電位掃描,從而研究電極反應的動力學和機理。掃描速率和濃度是影響循環伏安圖的重要因素
    的頭像 發表于 10-14 14:41 ?1804次閱讀

    功率半導體器件功率循環測試與控制策略

    功率循環測試是一種功率半導體器件的可靠性測試方法,被列為AEC-Q101與AQG-324等車規級測試標準內的必測項目。與溫度循環測試相比,功率循環是通過器件內部工作的芯片產生熱量,使得器件達到既定的溫度;而溫度
    的頭像 發表于 10-09 18:11 ?453次閱讀
    功率半導體器件功率<b class='flag-5'>循環</b>測試與控制策略

    rnn是遞歸神經網絡還是循環神經網絡

    RNN(Recurrent Neural Network)是循環神經網絡,而非遞歸神經網絡。循環神經網絡是一種具有時間序列特性的神經網絡,能夠處理序列數據,具有記憶功能。以下是關于循環神經網絡的介紹
    的頭像 發表于 07-05 09:52 ?625次閱讀

    循環神經網絡算法原理及特點

    循環神經網絡(Recurrent Neural Network,簡稱RNN)是一種具有記憶功能的神經網絡,能夠處理序列數據。與傳統的前饋神經網絡(Feedforward Neural Network
    的頭像 發表于 07-04 14:49 ?824次閱讀

    循環神經網絡的缺點是存在什么問題

    過程中,由于其循環結構,梯度在反向傳播時會經過多次乘法操作。這可能導致梯度在某些情況下變得非常小,即梯度消失問題,或者變得非常大,即梯度爆炸問題。這些問題會導致RNN的訓練過程變得非常困難,甚至無法收斂。 長期依賴問題 RNN的一個重要特點是能夠捕捉長距離的
    的頭像 發表于 07-04 14:41 ?1116次閱讀

    循環神經網絡的基本概念

    循環神經網絡(Recurrent Neural Network,簡稱RNN)是一種具有循環結構的神經網絡,其核心思想是將前一個時間步的輸出作為下一個時間步的輸入,從而實現對序列數據的建模。本文將從
    的頭像 發表于 07-04 14:31 ?773次閱讀

    CRC(循環冗余校驗)應用舉例

    CRC(循環冗余校驗)應用舉例
    的頭像 發表于 05-16 16:12 ?1398次閱讀

    什么是PLC循環移位指令 PLC循環移位的特點

    PLC循環移位指令包括循環左移指令和循環右移指令。在循環移位過程中,移出的位并不會丟失,而是會放回空出的位上,形成一個環形移位。
    的頭像 發表于 03-07 16:57 ?2275次閱讀
    什么是PLC<b class='flag-5'>循環</b>移位指令 PLC<b class='flag-5'>循環</b>移位的特點

    什么是RNN (循環神經網絡)?

    循環神經網絡 (RNN) 是一種深度學習結構,它使用過去的信息來提高網絡處理當前和將來輸入的性能。RNN 的獨特之處在于該網絡包含隱藏狀態和循環
    發表于 02-29 14:56 ?4151次閱讀
    什么是RNN (<b class='flag-5'>循環</b>神經網絡)?

    verilog中for循環是串行執行還是并行執行

    在Verilog中,for循環是并行執行的。Verilog是一種硬件描述語言,用于描述和設計數字電路和系統。在硬件系統中,各個電路模塊是同時運行的,并且可以并行執行多個操作。因此,在Verilog中
    的頭像 發表于 02-22 16:06 ?3080次閱讀

    arduino如何停止loop循環

    Arduino的loop循環是其主要的程序執行部分,該循環將在Arduino開發板上持續運行,并且只有在程序被重新上傳或開發板斷電重啟時才會停止。然而,在某些情況下,你可能需要在程序執行過程中停止或
    的頭像 發表于 02-14 16:24 ?4663次閱讀

    arduino中while循環怎么跳出

    Arduino 是一款開源的硬件平臺,廣泛應用于各種物聯網和嵌入式系統項目。在 Arduino 上編寫代碼時,循環結構起到了至關重要的作用。而其中的 while 循環更是常用于需要根據特定條件重復
    的頭像 發表于 02-14 16:22 ?2675次閱讀

    循環指令loop規定循環次數

    循環指令是計算機編程中非常重要的概念,它允許程序重復執行一段代碼塊,使得程序可以更有效地處理大量數據和重復性任務。在本文中,我們將詳盡、詳實、細致地介紹循環指令的相關概念、語法和應用場
    的頭像 發表于 02-14 16:10 ?1799次閱讀
    主站蜘蛛池模板: 99久久热视频只有精品| 韩国无遮羞禁动漫在线观看| 口工漫画r18全彩啪啪| 中文字幕精品无码一区二区| 年轻的母亲4线在线观看完整| 村上里沙快播| 国产A级毛片久久久久久久| 午夜伦理在线观看| 久久婷婷五月综合色情| 台湾佬休闲中性娱乐网| 国内精品乱码卡一卡2卡三卡| 5g天天影院天天看天天爽| 日日操夜夜操天天操| 狠狠色丁香婷婷久久综合| hd性欧美俱乐部中文| 欧美性受xxxx狂喷水| 国产精品久久久久久久久LI无码 | 动漫美女无衣| 一级黄色香蕉视频| 接吻吃胸摸下面啪啪教程| 9420高清免费观看在线大全| 我年轻漂亮的继坶2中字在线播放| 旧里番ovaの催○セイ活指导| 国产99久久久欧美黑人刘玥| 中文字幕精品在线视频| 无人区乱码1区2区3区网站| 久久婷婷电影网| 国产精品资源网站在线观看| gayxxxxgay呻吟受日本| 一区两区三不卡| 午夜国产视频| 泡妞高手在都市免费观看| 久久国产精品永久免费网站| 国产AV亚洲国产AV麻豆| 91情国产l精品国产亚洲区| 亚洲区视频| 午夜天堂AV久久久噜噜噜| 欧美视频 亚洲视频| 久久午夜伦理| 极品少妇小泬50PTHEPON| 国产精品久久人妻无码蜜|