【死磕 Spring】----- IOC 之 註冊 BeanDefinition

来源:https://www.cnblogs.com/chenssy/archive/2018/09/19/9672482.html
-Advertisement-
Play Games

原文出自: "http://cmsblogs.com" 獲取 Document 對象後,會根據該對象和 Resource 資源對象調用 方法,開始註冊 BeanDefinitions 之旅。如下: 首先調用 方法實例化 BeanDefinitionDocumentReader 對象,然後獲取統計前 ...


原文出自:http://cmsblogs.com

獲取 Document 對象後,會根據該對象和 Resource 資源對象調用 registerBeanDefinitions() 方法,開始註冊 BeanDefinitions 之旅。如下:

    public int registerBeanDefinitions(Document doc, Resource resource) throws BeanDefinitionStoreException {
        BeanDefinitionDocumentReader documentReader = createBeanDefinitionDocumentReader();
        int countBefore = getRegistry().getBeanDefinitionCount();
        documentReader.registerBeanDefinitions(doc, createReaderContext(resource));
        return getRegistry().getBeanDefinitionCount() - countBefore;
    }

首先調用 createBeanDefinitionDocumentReader() 方法實例化 BeanDefinitionDocumentReader 對象,然後獲取統計前 BeanDefinition 的個數,最後調用 registerBeanDefinitions() 註冊 BeanDefinition。

實例化 BeanDefinitionDocumentReader 對象方法如下:

    protected BeanDefinitionDocumentReader createBeanDefinitionDocumentReader() {
        return BeanDefinitionDocumentReader.class.cast(BeanUtils.instantiateClass(this.documentReaderClass));
    }

註冊 BeanDefinition 的方法 registerBeanDefinitions() 是在介面 BeanDefinitionDocumentReader 中定義,如下:

    void registerBeanDefinitions(Document doc, XmlReaderContext readerContext)
            throws BeanDefinitionStoreException;

從給定的 Document 對象中解析定義的 BeanDefinition 並將他們註冊到註冊表中。方法接收兩個參數,待解析的 Document 對象,以及解析器的當前上下文,包括目標註冊表和被解析的資源。其中 readerContext 是根據 Resource 來創建的,如下:

    public XmlReaderContext createReaderContext(Resource resource) {
        return new XmlReaderContext(resource, this.problemReporter, this.eventListener,
                this.sourceExtractor, this, getNamespaceHandlerResolver());
    }

DefaultBeanDefinitionDocumentReader 對該方法提供了實現:

    public void registerBeanDefinitions(Document doc, XmlReaderContext readerContext) {
        this.readerContext = readerContext;
        logger.debug("Loading bean definitions");
        Element root = doc.getDocumentElement();
        doRegisterBeanDefinitions(root);
    }

調用 doRegisterBeanDefinitions() 開啟註冊 BeanDefinition 之旅。

    protected void doRegisterBeanDefinitions(Element root) {
        BeanDefinitionParserDelegate parent = this.delegate;
        this.delegate = createDelegate(getReaderContext(), root, parent);

        if (this.delegate.isDefaultNamespace(root)) {
              // 處理 profile
            String profileSpec = root.getAttribute(PROFILE_ATTRIBUTE);
            if (StringUtils.hasText(profileSpec)) {
                String[] specifiedProfiles = StringUtils.tokenizeToStringArray(
                        profileSpec, BeanDefinitionParserDelegate.MULTI_VALUE_ATTRIBUTE_DELIMITERS);
                if (!getReaderContext().getEnvironment().acceptsProfiles(specifiedProfiles)) {
                    if (logger.isInfoEnabled()) {
                        logger.info("Skipped XML bean definition file due to specified profiles [" + profileSpec +
                                "] not matching: " + getReaderContext().getResource());
                    }
                    return;
                }
            }
        }

      // 解析前處理
        preProcessXml(root);
        // 解析
        parseBeanDefinitions(root, this.delegate);
        // 解析後處理
        postProcessXml(root);

        this.delegate = parent;
    }

程式首先處理 profile屬性,profile主要用於我們切換環境,比如切換開發、測試、生產環境,非常方便。然後調用 parseBeanDefinitions() 進行解析動作,不過在該方法之前之後分別調用 preProcessXml()postProcessXml() 方法來進行前、後處理,目前這兩個方法都是空實現,交由子類來實現。

    protected void preProcessXml(Element root) {
    }
    
    protected void postProcessXml(Element root) {
    }

parseBeanDefinitions() 定義如下:

    protected void parseBeanDefinitions(Element root, BeanDefinitionParserDelegate delegate) {
        if (delegate.isDefaultNamespace(root)) {
            NodeList nl = root.getChildNodes();
            for (int i = 0; i < nl.getLength(); i++) {
                Node node = nl.item(i);
                if (node instanceof Element) {
                    Element ele = (Element) node;
                    if (delegate.isDefaultNamespace(ele)) {
                        parseDefaultElement(ele, delegate);
                    }
                    else {
                        delegate.parseCustomElement(ele);
                    }
                }
            }
        }
        else {
            delegate.parseCustomElement(root);
        }
    }

最終解析動作落地在兩個方法處:parseDefaultElement(ele, delegate)delegate.parseCustomElement(root)。我們知道在 Spring 有兩種 Bean 聲明方式:

  • 配置文件式聲明:<bean id="studentService" class="org.springframework.core.StudentService"/>
  • 自定義註解方式:<tx:annotation-driven>

兩種方式的讀取和解析都存在較大的差異,所以採用不同的解析方法,如果根節點或者子節點採用預設命名空間的話,則調用 parseDefaultElement() 進行解析,否則調用 delegate.parseCustomElement() 方法進行自定義解析。

至此,doLoadBeanDefinitions() 中做的三件事情已經全部分析完畢,下麵將對 Bean 的解析過程做詳細分析說明。


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

-Advertisement-
Play Games
更多相關文章
  • 我們知道,js函數有多種寫法,函數聲明 ,函數表達式,Function式構造函數,自執行函數,包括Es6的箭頭函數,Class類寫法,高階函數,函數節流/函數防抖,下麵我就開始講關於上面幾種類型的最基本用法。 函數聲明式寫法 這種寫法是最基本的寫法 ,使用關鍵字 function 定義函數,函數聲明 ...
  • 半年前跳槽, 新公司主要研發傾向於小程式的開發。由於之前並沒有接觸小程式,所以經過半年的實際開發,才敢來做一點筆記。 小程式提供很多組件給開發者使用,但是,實際使用中還是會有很多的問題。 老生常談的不能使用npm開發??? (雖然,已知的很多小程式框架可以實現。例如:mpvue等。但是讓我們先討論討 ...
  • 在 JavaScript 中正確使用地使用 Array 的方法如下: 用 Array.includes 代替 Array.indexOf “如果你要在數組中查找元素,請使用 Array.indexOf”。 MDN 文檔寫道,Array.indexOf 將“返回第一次出現給定元素的索引”。因此,如果我 ...
  • 今天是我們介紹數組系列文章的第五篇,也是我們數組系列的最後一篇文章,只是數據系列的結束,所以大家不用擔心,我們會持續的更新乾貨文章。 生命不息,更新不止! 今天我們就不那麼多廢話了,直接乾貨開始。 我們在《Javascript數組系列一之棧與隊列》中描述我們是如何利用 push、pop、shift、 ...
  • 什麼是斷點操作(Breakpoint action) 做前端開發的小伙伴,或許對這個斷點操作不是很熟悉。不過你要是問其他語言(比如C,C++ ,C #等)的開發者,他們應該都挺熟悉斷點操作,這種斷點操作在諸如XCode或者Visual Studio的IDE中都會有提供。 以下一段話來自知乎 斷點操作 ...
  • JavaScript語言從設計之初就是考慮帶分號的,使用不帶分號的編碼規則就要小心點啦。 ...
  • 1、解決HTML中的編碼問題 2、列表的使用 3、HTML中的表格標記 4、表單標記 ...
  • 一.Bootstrap 概述Bootstrap 是由 Twitter 公司(全球最大的微博)的兩名技術工程師研發的一個基於HTML、CSS、JavaScript 的開源框架。該框架代碼簡潔、視覺優美,可用於快速、簡單地構建基於 PC 及移動端設備的 Web 頁面需求。2010 年 6 月,Twitt ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...