Spring 源碼(14)Spring Bean 的創建過程(6)對象的提前暴露

来源:https://www.cnblogs.com/redwinter/archive/2022/05/18/16286745.html
-Advertisement-
Play Games

知識回顧 解析完Bean信息的合併,可以知道Spring在實例化Bean之後,屬性填充前,對Bean進行了Bean的合併操作,這裡的操作主要做了對Bean對象標記了@Autowired、@Value、@Resource、@PostConstruct、@PreDestroy註解的欄位或者方法進行解析, ...


知識回顧

解析完Bean信息的合併,可以知道Spring在實例化Bean之後,屬性填充前,對Bean進行了Bean的合併操作,這裡的操作主要做了對Bean對象標記了@Autowired@Value@Resource@PostConstruct@PreDestroy註解的欄位或者方法進行解析,主要涉及到類都是BeanPostProcessor的實現,可見BeanPostProcessor介面的重要性。

這裡再次回顧下BeanPostProcessor介面有哪些子介面:

  • InstantiationAwareBeanPostProcessor

    Spring 給機會提前進行實例化,可用通過代理進行對象的創建

  • SmartInstantiationAwareBeanPostProcessor

    用於預測Bean的類型,決定Bean的構造函數,用於實例化

  • MergedBeanDefinitionPostProcessor

    用於合併Bean的信息,即解析Bean對象方法上或者欄位上標記的註解,比如@Resource@Autowired@PostConstruct等。

  • DestructionAwareBeanPostProcessor

    用於銷毀Bean時調用的,比如執行標有@PreDestroy註解的方法

這些子介面的實現類比較多,比如:

  • AutowiredAnnotationBeanPostProcessor
  • ComonAnnotationBeanPostProcessor
  • InitDestroyAnnotationBeanPostProcessor
  • AnnotationAwareAspectJAutoProxyCreator
  • ScheduledAnnotationBeanPostProcessor

當然還不止這裡列出來的,還有其他的就不列了,接下來分析Spring源碼接下來做了什麼?

對象的提前暴露

看源碼:

// 省略代碼....
// Eagerly cache singletons to be able to resolve circular references
// even when triggered by lifecycle interfaces like BeanFactoryAware.
// 提前暴露對象,用於解決迴圈依賴
boolean earlySingletonExposure = (mbd.isSingleton() && this.allowCircularReferences &&
                                  isSingletonCurrentlyInCreation(beanName));
if (earlySingletonExposure) {
  if (logger.isTraceEnabled()) {
    logger.trace("Eagerly caching bean '" + beanName +
                 "' to allow for resolving potential circular references");
  }
  // 添加一個lambda表達式到三級緩存中
  addSingletonFactory(beanName, () -> getEarlyBeanReference(beanName, mbd, bean));
}

// Initialize the bean instance.
Object exposedObject = bean;
// 省略代碼....

源碼這裡就添加了一個lambda表達式到一個Map中,然後結束了,並且明確說明瞭提前暴露是為瞭解決迴圈依賴問題。

什麼是迴圈依賴?

迴圈依賴顧名思義,就是你中有我,我中有你,打個比方現在有個對象A,他有個屬性b,這個屬性b是對象B的,然後對象B中有個屬性a,屬性a是對象A的。

現在開始創建對象,按照Spring的標準創建流程getBean-->doGetBean-->createBean-->doCreateBean,先實例化,然後屬性填充,然後執行aware方法,然後執行BeanPostProcessorbefore方法,然後執行init-method,然後執行BeanPostProcessorafter方法。那麼在執行屬性填充時必然會去查找a或者b屬性對應的對象,如果找不到就會去創建,那麼就會出現下圖的樣子:

這樣必然就出現了迴圈依賴,你我緊緊相擁,不想放開,死也要在一起的情形。

那麼Spring為什麼解決迴圈依賴需要進行提前暴露對象呢?

所以這個問題就很簡單了,我們都知道Bean的創建是將實例化和初始化分開的,實例化之後的對象在JVM堆中已經開闢了記憶體空間地址,這個地址是不會變的,除非山崩地裂,海枯石爛,也就是應用重啟了。

因此可以將已經實例化的對象放在另外一個Map中,一般來說都稱之為半成品,當填充屬性時,可以將先設置半成品對象,等到對象創建完之後在將半成品換成成品,這樣的話對象進行屬性填充時就可以直接先使用半成品填充,等到開始初始化時再將對象創建出來即可。

這樣看來迴圈依賴只需要二級緩存就夠了,但是在Spring中,存在一種特殊的對象,就是代理對象。也就是說在放入的半成品我們現在多了一種對象,那就是代理對象,這個時候就會出現使用代理對象還是普通對象呢?所以乾脆在搞一個Map專門存放代理對象,這樣就區分出來了,然後在使用的時候先判斷下我們創建的對象是需要代理還是不需要代理,如果需要代理,那麼就創建一個代理對象放在map中,否則直接使用普通對象就可以了。

Spring中的實現方式

Spring是怎麼處理的呢?Spring是將所有的對象都放在三級緩存中,也就是lambda表達式中:

// 添加一個lambda表達式到三級緩存中
addSingletonFactory(beanName, () -> getEarlyBeanReference(beanName, mbd, bean));


protected void addSingletonFactory(String beanName, ObjectFactory<?> singletonFactory) {
  Assert.notNull(singletonFactory, "Singleton factory must not be null");
  synchronized (this.singletonObjects) {
    // 判斷一級緩存中是否存在
    if (!this.singletonObjects.containsKey(beanName)) {
      // 沒有就放入三級緩存中
      this.singletonFactories.put(beanName, singletonFactory);
      // 清空二級緩存
      this.earlySingletonObjects.remove(beanName);
      // 添加到已經註冊的單例集合中
      this.registeredSingletons.add(beanName);
    }
  }
}

在屬性填充的時候,會執行到getBean,然後從緩存中獲取getSingleton

protected Object getSingleton(String beanName, boolean allowEarlyReference) {
  // Quick check for existing instance without full singleton lock
  // 從一級緩存中獲取bean實例
  Object singletonObject = this.singletonObjects.get(beanName);
  // 如果一級緩存中沒有數據並且沒有正在創建的Bean直接返回
  if (singletonObject == null && isSingletonCurrentlyInCreation(beanName)) {
    // 如果有正在創建的Bean,那麼沖二級緩存中獲取,早期的單例對象
    singletonObject = this.earlySingletonObjects.get(beanName);
    if (singletonObject == null && allowEarlyReference) {
      synchronized (this.singletonObjects) {
        // Consistent creation of early reference within full singleton lock
        // 二次檢查一級緩存中是否有單例對象
        singletonObject = this.singletonObjects.get(beanName);
        if (singletonObject == null) {
          // 二次判斷二級緩存中是否存在單例對象
          singletonObject = this.earlySingletonObjects.get(beanName);
          if (singletonObject == null) {
            // 從三級緩存中獲取Bean
            ObjectFactory<?> singletonFactory = this.singletonFactories.get(beanName);
            if (singletonFactory != null) {
              // 如果三級緩存中 單例工廠中有對象,那麼就將該對象放在二級緩存中,並且清掉三級緩存
              singletonObject = singletonFactory.getObject();
              this.earlySingletonObjects.put(beanName, singletonObject);
              this.singletonFactories.remove(beanName);
            }
          }
        }
      }
    }
  }
  return singletonObject;
}

在獲取單例對象時,會執行到三級緩存,然後執行getObject方法,最終就會觸發getEarlyBeanReference方法的調用:

protected Object getEarlyBeanReference(String beanName, RootBeanDefinition mbd, Object bean) {
  Object exposedObject = bean;
  // 添加 三級緩存,判斷是否需要進行代理創建對象,是一個動態代理創建的代理對象
  if (!mbd.isSynthetic() && hasInstantiationAwareBeanPostProcessors()) {
    for (BeanPostProcessor bp : getBeanPostProcessors()) {
      if (bp instanceof SmartInstantiationAwareBeanPostProcessor) {
        SmartInstantiationAwareBeanPostProcessor ibp = (SmartInstantiationAwareBeanPostProcessor) bp;
        exposedObject = ibp.getEarlyBeanReference(exposedObject, beanName);
      }
    }
  }
  return exposedObject;
}

這裡會判斷,如果這個BeanDefinition是否滿足條件,如果不滿足,那麼直接返回了,否則就會執行到for迴圈中的代碼,而getEarlyBeanReference方法在Spring中只有AbstractAutoProxyCreator類進行了實質的實現:

public Object getEarlyBeanReference(Object bean, String beanName) {
  Object cacheKey = getCacheKey(bean.getClass(), beanName);
  // 早期代理對象的引用集合
  this.earlyProxyReferences.put(cacheKey, bean);
  // 創建代理
  return wrapIfNecessary(bean, beanName, cacheKey);
}

點進去:

protected Object wrapIfNecessary(Object bean, String beanName, Object cacheKey) {
  if (StringUtils.hasLength(beanName) && this.targetSourcedBeans.contains(beanName)) {
    return bean;
  }
  if (Boolean.FALSE.equals(this.advisedBeans.get(cacheKey))) {
    return bean;
  }
  if (isInfrastructureClass(bean.getClass()) || shouldSkip(bean.getClass(), beanName)) {
    this.advisedBeans.put(cacheKey, Boolean.FALSE);
    return bean;
  }

  // Create proxy if we have advice.
  Object[] specificInterceptors = getAdvicesAndAdvisorsForBean(bean.getClass(), beanName, null);
  if (specificInterceptors != DO_NOT_PROXY) {
    this.advisedBeans.put(cacheKey, Boolean.TRUE);
    // 創建代理對象
    Object proxy = createProxy(
      bean.getClass(), beanName, specificInterceptors, new SingletonTargetSource(bean));
    this.proxyTypes.put(cacheKey, proxy.getClass());
    return proxy;
  }

  this.advisedBeans.put(cacheKey, Boolean.FALSE);
  return bean;
}

首先進行了判斷,如果不滿足創建代理的條件,都是直接返回這個對象,否則進入創建代理的方法,創建出代理對象,最終放入緩存中。點入到最後會發現使用了兩種代理創建方式:

源碼中的提前暴露對象牽扯出很多東西,迴圈依賴,三級緩存,aop等,這裡解析了個大概,接下來繼續主流程中的屬性填充populaeBean方法。


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

-Advertisement-
Play Games
更多相關文章
  • 在產品運營的工作過程中,需要每日關註產品的核心指標變化情況,監控其整體運營狀況。華為分析服務提供查看吸引新用戶卡片,該卡片展示了新增用戶數、人均會話次數、人均訪問時長、人均頁面訪問數。藉助該頁面運營可觀察拉新效果,判斷產品對新用戶的吸引力。 問題描述 某開發者在集成華為分析服務後,發現AGC概覽頁面 ...
  • 現在正式回歸,開始好好做項目了,正好這一個項目也開始慢慢的開始起色了,前面的準備工作都做的差不多了。 而且我現在也開始慢慢瞭解到了一些項目才開始需要的一些什麼東西了,vuex、router這些都是必備的,後期一定要練得非常熟練才行。 一.重寫push/replace方法 有一個編程式導航的bug,當 ...
  • 常用事件 onload <script> window.onload = function () { ele = document.getElementById("i") console.log(ele.innerHTML); } </script> </head> <body> <div clas ...
  • DOM DOM document Object Model 文檔對象模型 // 整個html文檔,會保存一個文檔對象document // console.log( document ); // 獲取當前文檔的對象 查找標簽 直接查找 document.getElementsByTagName("標 ...
  • 在 CSS 選擇器家族中,新增這樣一類比較新的選擇器 -- 邏輯選擇器,目前共有 4 名成員: :is :where :not :has 本文將帶領大家瞭解、深入它們。做到學以致用,寫出更現代化的選擇器。 :is 偽類選擇器 :is() CSS偽類函數將選擇器列表作為參數,並選擇該列表中任意一個選擇 ...
  • 前端處理二進位流數據--轉下載 導言 ​ 因業務需要,實現分類導出功能。篩選導出一定條件的數據,後端處理成Excle數據流,前端實現導出下載。 實現 方法一 ​ 將條件格式化成key=value&...文本格式,接到<a>標簽url介面之後,每當點擊導出按鈕的時候,創建一個<a>標簽,寫入介面地址+ ...
  • 本章是系列文章的第三章,介紹了基於數據流分析的一些優化方法。包括生命周期管理,可獲得表達式,常用表達式,可達性定義。本章在介紹這4中分析方法的基礎上提取出它們的通用模式。這一章形式化的內容比較多,看的時候有點燒腦,最好自己手工推導一下,要不然基本上看不懂:) 本文中的所有內容來自學習DCC888的學 ...
  • 一、吐槽 已經是凌晨12點了我還是睡不著 我所有的實體類時間用的j8的LocalDateTime 這就導致一個問題:jackson不能序列化時間,因為它不支持j8的Api,讓我添加 jackson-datatype-jsr310 解決 二、問題 如果是這樣做統一返回結果集需要 private sta ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...