【死磕 Spring】----- IOC 之解析 bean 標簽:開啟解析進程

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

原文出自: "http://cmsblogs.com" import 標簽解析完畢了,再看 Spring 中最複雜也是最重要的標簽 bean 標簽的解析過程。 在方法 中,如果遇到標簽 為 bean 則調用 方法進行 bean 標簽解析,如下: 整個過程分為四個步驟 1. 調用 進行元素解析,解析過 ...


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

import 標簽解析完畢了,再看 Spring 中最複雜也是最重要的標簽 bean 標簽的解析過程。

在方法 parseDefaultElement() 中,如果遇到標簽 為 bean 則調用 processBeanDefinition() 方法進行 bean 標簽解析,如下:

    protected void processBeanDefinition(Element ele, BeanDefinitionParserDelegate delegate) {
        BeanDefinitionHolder bdHolder = delegate.parseBeanDefinitionElement(ele);
        if (bdHolder != null) {
            bdHolder = delegate.decorateBeanDefinitionIfRequired(ele, bdHolder);
            try {
                // Register the final decorated instance.
                BeanDefinitionReaderUtils.registerBeanDefinition(bdHolder, getReaderContext().getRegistry());
            }
            catch (BeanDefinitionStoreException ex) {
                getReaderContext().error("Failed to register bean definition with name '" +
                        bdHolder.getBeanName() + "'", ele, ex);
            }
            // Send registration event.
            getReaderContext().fireComponentRegistered(new BeanComponentDefinition(bdHolder));
        }
    }

整個過程分為四個步驟

  1. 調用 BeanDefinitionParserDelegate.parseBeanDefinitionElement() 進行元素解析,解析過程中如果失敗,返回 null,錯誤由 ProblemReporter 處理。如果解析成功則返回 BeanDefinitionHolder 實例 bdHolder。BeanDefinitionHolder 為持有 name 和 alias 的 BeanDefinition。
  2. 若實例 bdHolder 不為空,則調用 BeanDefinitionParserDelegate.decorateBeanDefinitionIfRequired() 進行自定義標簽處理
  3. 解析完成後,則調用 BeanDefinitionReaderUtils.registerBeanDefinition() 對 bdHolder 進行註冊
  4. 發出響應事件,通知相關的監聽器,完成 Bean 標簽解析

先看方法 parseBeanDefinitionElement(),如下:

   public BeanDefinitionHolder parseBeanDefinitionElement(Element ele, @Nullable BeanDefinition containingBean) {
        // 解析 ID 屬性
        String id = ele.getAttribute(ID_ATTRIBUTE);
        // 解析 name 屬性
        String nameAttr = ele.getAttribute(NAME_ATTRIBUTE);

        // 分割 name 屬性
        List<String> aliases = new ArrayList<>();
        if (StringUtils.hasLength(nameAttr)) {
            String[] nameArr = StringUtils.tokenizeToStringArray(nameAttr, MULTI_VALUE_ATTRIBUTE_DELIMITERS);
            aliases.addAll(Arrays.asList(nameArr));
        }
    
        String beanName = id;
        if (!StringUtils.hasText(beanName) && !aliases.isEmpty()) {
            beanName = aliases.remove(0);
            if (logger.isDebugEnabled()) {
                logger.debug("No XML 'id' specified - using '" + beanName +
                        "' as bean name and " + aliases + " as aliases");
            }
        }
        
        // 檢查 name 的唯一性
        if (containingBean == null) {
            checkNameUniqueness(beanName, aliases, ele);
        }

        // 解析 屬性,構造 AbstractBeanDefinition
        AbstractBeanDefinition beanDefinition = parseBeanDefinitionElement(ele, beanName, containingBean);
        if (beanDefinition != null) {
            // 如果 beanName 不存在,則根據條件構造一個 beanName
            if (!StringUtils.hasText(beanName)) {
                try {
                    if (containingBean != null) {
                        beanName = BeanDefinitionReaderUtils.generateBeanName(
                                beanDefinition, this.readerContext.getRegistry(), true);
                    }
                    else {
                        beanName = this.readerContext.generateBeanName(beanDefinition);
                        String beanClassName = beanDefinition.getBeanClassName();
                        if (beanClassName != null &&
                                beanName.startsWith(beanClassName) && beanName.length() > beanClassName.length() &&
                                !this.readerContext.getRegistry().isBeanNameInUse(beanClassName)) {
                            aliases.add(beanClassName);
                        }
                    }
                    if (logger.isDebugEnabled()) {
                        logger.debug("Neither XML 'id' nor 'name' specified - " +
                                "using generated bean name [" + beanName + "]");
                    }
                }
                catch (Exception ex) {
                    error(ex.getMessage(), ele);
                    return null;
                }
            }
            String[] aliasesArray = StringUtils.toStringArray(aliases);
            
            // 封裝 BeanDefinitionHolder
            return new BeanDefinitionHolder(beanDefinition, beanName, aliasesArray);
        }

        return null;
    }

這個方法還沒有對 Bean 標簽進行解析,只是在解析動作之前做了一些功能架構,主要的工作有:

  • 解析 id、name 屬性,確定 alias 集合,檢測 beanName 是否唯一
  • 調用方法 parseBeanDefinitionElement() 對屬性進行解析並封裝成 GenericBeanDefinition 實例 beanDefinition
  • 根據所獲取的信息(beanName、aliases、beanDefinition)構造 BeanDefinitionHolder 實例對象並返回。

這裡有必要說下 beanName 的命名規則:如果 id 不為空,則 beanName = id;如果 id 為空,但是 alias 不空,則 beanName 為 alias 的第一個元素,如果兩者都為空,則根據預設規則來設置 beanName。

上面三個步驟第二個步驟為核心方法,它主要承擔解析 Bean 標簽中所有的屬性值。如下:

   public AbstractBeanDefinition parseBeanDefinitionElement(
            Element ele, String beanName, @Nullable BeanDefinition containingBean) {

        this.parseState.push(new BeanEntry(beanName));

        String className = null;
        // 解析 class 屬性
        if (ele.hasAttribute(CLASS_ATTRIBUTE)) {
            className = ele.getAttribute(CLASS_ATTRIBUTE).trim();
        }
        String parent = null;

        // 解析 parent 屬性
        if (ele.hasAttribute(PARENT_ATTRIBUTE)) {
            parent = ele.getAttribute(PARENT_ATTRIBUTE);
        }

        try {

            // 創建用於承載屬性的 GenericBeanDefinition 實例
            AbstractBeanDefinition bd = createBeanDefinition(className, parent);

            // 解析預設 bean 的各種屬性
            parseBeanDefinitionAttributes(ele, beanName, containingBean, bd);
            
            // 提取 description
            bd.setDescription(DomUtils.getChildElementValueByTagName(ele, DESCRIPTION_ELEMENT));
            
            // 解析元數據
            parseMetaElements(ele, bd);
            
            // 解析 lookup-method 屬性
            parseLookupOverrideSubElements(ele, bd.getMethodOverrides());
            
            // 解析 replaced-method 屬性
            parseReplacedMethodSubElements(ele, bd.getMethodOverrides());
            
            // 解析構造函數參數
            parseConstructorArgElements(ele, bd);
            
            // 解析 property 子元素
            parsePropertyElements(ele, bd);
            
            // 解析 qualifier 子元素
            parseQualifierElements(ele, bd);

            bd.setResource(this.readerContext.getResource());
            bd.setSource(extractSource(ele));

            return bd;
        }
        catch (ClassNotFoundException ex) {
            error("Bean class [" + className + "] not found", ele, ex);
        }
        catch (NoClassDefFoundError err) {
            error("Class that bean class [" + className + "] depends on not found", ele, err);
        }
        catch (Throwable ex) {
            error("Unexpected failure during bean definition parsing", ele, ex);
        }
        finally {
            this.parseState.pop();
        }

        return null;
    }

到這裡,Bean 標簽的所有屬性我們都可以看到其解析的過程,也就說到這裡我們已經解析一個基本可用的 BeanDefinition。

由於解析過程較為漫長,篇幅較大,為了更好的觀看體驗,將這篇博文進行拆分。下篇博客主要介紹 BeanDefinition,以及解析預設 Bean 的過程(parseBeanDefinitionAttributes()

--


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

-Advertisement-
Play Games
更多相關文章
  • 距離上次發文,已經有一段時間了,最近工作比較忙,這不眼看快雙十一了,就相當於給大家一些福利吧! 一、什麼是數組去重 簡單說就是把數組中重覆的項刪除掉,你 GET 到了嗎 ?下麵我將簡單介紹下幾種基本的方法及其優缺點。 二、方法彙總 兩層迴圈 無相同值直接 進新數組,有相同的值則直接跳過本次內部迴圈 ...
  • 描述: 本文主要是講,通過css+js實現網頁中的【返回頂部】功能。 實現代碼: HTML: CSS: JS: ...
  • 簡稱 js防連點 var flag = true; $(".yzm>span").click(function(){ if(!flag){ return false } flag = false; var time = 60; var timer = setInterval(function(){ ...
  • 前段時間,公司在項目上用到了xhEditor編輯器來給用戶做一個上傳圖片的功能當時做的時候覺得很有意思,想想 基本的用戶圖片上傳到自己伺服器,還有點小占地方; 後來....然後直接上傳到阿裡雲 。接下來就是基本操作: 首先,引入官方提供的js庫 註:xhEditor插件下載官網:https://xh ...
  • TSAD的來源: TSAD由Open-test、Open-stor、Open-api、Open-dev四大系統組成,提供API測試平臺Open-test;測試通過版本可發佈服務倉庫Open-stor,倉庫抽離單一服務,其他產品部可任意裝配服務;根據業務需求將服務開放到Open-api與Open-de ...
  • 教程:高能:語句結構都是由關鍵字開頭,用冒號結束! 一:語句結構for <variable> in <sequence>: <statements>else: # else可有可無 <statements>二:基本規則 (1)使用縮進來劃分語句塊,相同縮進數的語句在一起組成一個語句塊。 (2)seq ...
  • ![](https://img2018.cnblogs.com/blog/711958/201809/711958-20180928091826555-1354813331.jpg) ...
  • ![](https://img2018.cnblogs.com/blog/711958/201809/711958-20180928091434379-573436458.jpg) ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...