欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Spring中的NamespaceHandler加載過(guò)程源碼詳解

 更新時(shí)間:2024年02月14日 08:13:06   作者:securitit  
這篇文章主要介紹了Spring中的NamespaceHandler加載過(guò)程源碼詳解,Spring提供的NamespaceHandler的處理機(jī)制,簡(jiǎn)單來(lái)說(shuō)就是命名空間處理器,Spring為了開(kāi)放性提供了NamespaceHandler機(jī)制,這樣我們就可以根據(jù)需求自己來(lái)處理我們?cè)O(shè)置的標(biāo)簽元素,需要的朋友可以參考下

NamespaceHandler加載過(guò)程流程

在這里插入圖片描述

流程圖中羅列了Spring加載配置的整個(gè)過(guò)程,其中,從BeanDefinitionParserDelegate.parseCustomElement(...)開(kāi)始,即開(kāi)始了對(duì)命名空間NamespaceHandler的解析,本文對(duì)這部分內(nèi)容進(jìn)行解析。

NamespaceHandler加載過(guò)程解析

1) BeanDefinitionParserDelegate.parseCustomElement(…)方法

① 取得指定元素的命名空間。

② 通過(guò)DefaultNamespaceHandlerResolver解析指定元素命名空間對(duì)應(yīng)的NamespaceHandler。

③ 通過(guò)查找的NamespaceHandler解析指定元素,取得Bean定義實(shí)例。

/**
 * 解析自定義元素.
 * @param ele 元素.
 * @param containingBd Bean定義.
 * @return 解析Bean定義.
 */
@Nullable
public BeanDefinition parseCustomElement(Element ele, @Nullable BeanDefinition containingBd) {
    // 獲取元素命名空間.
    String namespaceUri = getNamespaceURI(ele);
    if (namespaceUri == null) {
        return null;
    }
    // 解析元素命名空間對(duì)應(yīng)的NamespaceHandler.
    NamespaceHandler handler = this.readerContext.getNamespaceHandlerResolver().resolve(namespaceUri);
    if (handler == null) {
        error("Unable to locate Spring NamespaceHandler for XML schema namespace [" + namespaceUri + "]", ele);
        return null;
    }
    // 解析當(dāng)前元素.
    return handler.parse(ele, new ParserContext(this.readerContext, this, containingBd));
}

2) DefaultNamespaceHandlerResolver.resolve(…)方法

① 加載所有依賴的META-INF/spring.handlers文件,取得命名空間對(duì)應(yīng)處理器映射。

② 根據(jù)命名空間從①的映射中查找對(duì)應(yīng)的NamespaceHandler類。

③ 映射配置的NamespaceHandler類必須實(shí)現(xiàn)NamespaceHandler接口,否則拋出異常。

④ 根據(jù)配置映射的NamespaceHandler類路徑進(jìn)行實(shí)例化。

⑤ 調(diào)用NamespaceHandler的init()方法,指定命名空間元素對(duì)應(yīng)的解析器。

/**
 * 從配置映射中查找所提供命名空間URI的NamespaceHandler.
 * @param namespaceUri 相關(guān)命名空間URI.
 * @return NamespaceHandler,若未找到,則為null.
 */
@Override
@Nullable
public NamespaceHandler resolve(String namespaceUri) {
    // 加載處理器映射.
    Map<String, Object> handlerMappings = getHandlerMappings();
    // 根據(jù)namespaceUri查找處理器類路徑.
    Object handlerOrClassName = handlerMappings.get(namespaceUri);
    if (handlerOrClassName == null) {
        return null;
    }
    else if (handlerOrClassName instanceof NamespaceHandler) {
        return (NamespaceHandler) handlerOrClassName;
    }
    else {
        String className = (String) handlerOrClassName;
        try {
            Class<?> handlerClass = ClassUtils.forName(className, this.classLoader);
            // 處理器需實(shí)現(xiàn)NamespaceHandler接口.
            if (!NamespaceHandler.class.isAssignableFrom(handlerClass)) {
                throw new FatalBeanException("Class [" + className + "] for namespace [" + namespaceUri +
                                             "] does not implement the [" + NamespaceHandler.class.getName() + "] interface");
            }
            NamespaceHandler namespaceHandler = (NamespaceHandler) BeanUtils.instantiateClass(handlerClass);
            // NamespaceHandler初始化,指定命名空間下各元素對(duì)應(yīng)的解析器.
            namespaceHandler.init();
            handlerMappings.put(namespaceUri, namespaceHandler);
            return namespaceHandler;
        }
        catch (ClassNotFoundException ex) {
            throw new FatalBeanException("Could not find NamespaceHandler class [" + className +
                                         "] for namespace [" + namespaceUri + "]", ex);
        }
        catch (LinkageError err) {
            throw new FatalBeanException("Unresolvable class definition for NamespaceHandler class [" +
                                         className + "] for namespace [" + namespaceUri + "]", err);
        }
    }
}

3) DefaultNamespaceHandlerResolver.getHandlerMappings()方法

① 加載所有依賴的META-INF/spring.handlers文件,取得命名空間對(duì)應(yīng)處理器映射,存儲(chǔ)在Properties實(shí)例中。

② 將數(shù)據(jù)由Properties轉(zhuǎn)存到Map中。

/**
 * 延遲加載指定的NamespaceHandler映射.
 */
private Map<String, Object> getHandlerMappings() {
    Map<String, Object> handlerMappings = this.handlerMappings;
    if (handlerMappings == null) {
        synchronized (this) {
            handlerMappings = this.handlerMappings;
            if (handlerMappings == null) {
                try {
                    // 搜索META-INF/spring.handlers文件.
                    Properties mappings =
                        PropertiesLoaderUtils.loadAllProperties(this.handlerMappingsLocation, this.classLoader);
                    if (logger.isDebugEnabled()) {
                        logger.debug("Loaded NamespaceHandler mappings: " + mappings);
                    }
                    Map<String, Object> mappingsToUse = new ConcurrentHashMap<>(mappings.size());
                    // 將Properties數(shù)據(jù)轉(zhuǎn)換到Map中.
                    CollectionUtils.mergePropertiesIntoMap(mappings, mappingsToUse);
                    handlerMappings = mappingsToUse;
                    this.handlerMappings = handlerMappings;
                }
                catch (IOException ex) {
                    throw new IllegalStateException(
                        "Unable to load NamespaceHandler mappings from location [" + this.handlerMappingsLocation + "]", ex);
                }
            }
        }
    }
    return handlerMappings;
}

4) NamespaceHandlerSupport.parse(…)方法

① 使用提供元素的本地名稱查找BeanDefinitionParser。

② 調(diào)用BeanDefinitionParser實(shí)現(xiàn)進(jìn)行解析。

/**
 * 通過(guò)委托給為Element注冊(cè)的BeanDefinitionParser來(lái)解析提供的Element.
 */
@Override
@Nullable
public BeanDefinition parse(Element element, ParserContext parserContext) {
    // 使用提供元素的本地名稱查找BeanDefinitionParser.
    BeanDefinitionParser parser = findParserForElement(element, parserContext);
    // 調(diào)用BeanDefinitionParser實(shí)現(xiàn)進(jìn)行解析.
    return (parser != null ? parser.parse(element, parserContext) : null);
}

5) NamespaceHandlerSupport.findParserForElement(…)方法

① 通過(guò)指定元素,取得元素的本地參數(shù)名稱。

② 根據(jù)本地名稱查找已注冊(cè)的BeanDefinitionParser。

/**
 * 使用提供的元素的本地名稱查找BeanDefinitionParser.
 */
@Nullable
private BeanDefinitionParser findParserForElement(Element element, ParserContext parserContext) {
    // 獲取提供元素的本地名稱.
    String localName = parserContext.getDelegate().getLocalName(element);
    // 根據(jù)本地名稱查找BeanDefinitionParser.
    BeanDefinitionParser parser = this.parsers.get(localName);
    if (parser == null) {
        parserContext.getReaderContext().fatal(
            "Cannot locate BeanDefinitionParser for element [" + localName + "]", element);
    }
    return parser;
}

6) BeanDefinitionParser.parse(…)方法

BeanDefinitionParser.parse(...)是由具體開(kāi)發(fā)進(jìn)行編寫的,其中涉及元素解析、元素校驗(yàn)、注冊(cè)對(duì)應(yīng)Beam。

總結(jié)

‘工欲善其事必先利其器’,只有當(dāng)對(duì)原理的了解達(dá)到一定程度,才能在實(shí)際遇到問(wèn)題,簡(jiǎn)潔、快速、高效的解決問(wèn)題。

源碼解析基于spring-framework-5.0.5.RELEASE版本源碼。

若文中存在錯(cuò)誤和不足,歡迎指正!

到此這篇關(guān)于Spring中的NamespaceHandler加載過(guò)程源碼詳解的文章就介紹到這了,更多相關(guān)NamespaceHandler加載過(guò)程內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • Mybatis的update更新批量與普通解決方式對(duì)比

    Mybatis的update更新批量與普通解決方式對(duì)比

    這篇文章主要為大家介紹了Mybatis的update更新批量與普通解決方式對(duì)比,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-04-04
  • 一篇超詳細(xì)的SpringBoot整合MybatisPlus的文章

    一篇超詳細(xì)的SpringBoot整合MybatisPlus的文章

    這篇文章主要介紹了springboot整合Mybatis-plus的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2021-07-07
  • java高并發(fā)InterruptedException異常引發(fā)思考

    java高并發(fā)InterruptedException異常引發(fā)思考

    這篇文章主要為大家介紹了java高并發(fā)InterruptedException異常引發(fā)思考,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-08-08
  • 詳解 Java 中 equals 和 == 的區(qū)別

    詳解 Java 中 equals 和 == 的區(qū)別

    這篇文章主要介紹了詳解 Java 中 equals 和 == 的區(qū)別的相關(guān)資料,equals 和 == 都是用來(lái)檢測(cè)兩個(gè)字符串是否相等,返回值也都是布爾型,但是兩者在內(nèi)部比較的處理中卻不盡相同需要的朋友可以參考下
    2017-03-03
  • microlog4android將Android Log日志寫到SD卡文件中實(shí)現(xiàn)方法

    microlog4android將Android Log日志寫到SD卡文件中實(shí)現(xiàn)方法

    這篇文章主要介紹了microlog4android將Android Log日志寫到SD卡文件中實(shí)現(xiàn)方法的相關(guān)資料,需要的朋友可以參考下
    2016-10-10
  • 解決Mybatis-plus和pagehelper依賴沖突的方法示例

    解決Mybatis-plus和pagehelper依賴沖突的方法示例

    這篇文章主要介紹了解決Mybatis-plus和pagehelper依賴沖突的方法示例,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2021-04-04
  • Java中一個(gè)線程執(zhí)行死循環(huán)有什么后果

    Java中一個(gè)線程執(zhí)行死循環(huán)有什么后果

    這篇文章主要為大家詳細(xì)介紹了Java中一個(gè)線程執(zhí)行死循環(huán)有什么后果,當(dāng)一個(gè)線程在執(zhí)行死循環(huán)時(shí)會(huì)影響另外一個(gè)線程嗎,下面為大家揭曉
    2016-05-05
  • ArrayList刪除集合中某一屬性相同的元素方法(推薦)

    ArrayList刪除集合中某一屬性相同的元素方法(推薦)

    下面小編就為大家?guī)?lái)一篇ArrayList刪除集合中某一屬性相同的元素方法(推薦)。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧
    2016-09-09
  • Jmeter線程組傳參原理解析

    Jmeter線程組傳參原理解析

    這篇文章主要介紹了jmeter線程組傳參原理解析,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2020-02-02
  • SpringBoot自動(dòng)配置源碼深入刨析講解

    SpringBoot自動(dòng)配置源碼深入刨析講解

    這篇文章主要介紹了SpringBoot自動(dòng)配置原理分析,SpringBoot是我們經(jīng)常使用的框架,那么你能不能針對(duì)SpringBoot實(shí)現(xiàn)自動(dòng)配置做一個(gè)詳細(xì)的介紹。如果可以的話,能不能畫一下實(shí)現(xiàn)自動(dòng)配置的流程圖。牽扯到哪些關(guān)鍵類,以及哪些關(guān)鍵點(diǎn)
    2022-09-09

最新評(píng)論