spring xml的bean提取 源碼學(xué)習(xí)

上篇Spring xml解析 源碼學(xué)習(xí)已經(jīng)細(xì)說(shuō)了spring是如何解析xml文件的哑舒,現(xiàn)在就來(lái)學(xué)習(xí)下如何把xml的元素轉(zhuǎn)換為beandefinition甚至于我們需要的bean妇拯。

從DefaultBeanDefinitionDocumentReader的parseBeanDefinitions函數(shù)開(kāi)始解析得出具體的bean,重點(diǎn)分析是默認(rèn)的命名空間洗鸵。
從root根節(jié)點(diǎn)出發(fā)越锈,循環(huán)遍歷其下的所有子節(jié)點(diǎn),針對(duì)每個(gè)子節(jié)點(diǎn)進(jìn)行判斷和處理膘滨。

protected void parseBeanDefinitions(Element root, BeanDefinitionParserDelegate delegate) {
    if (delegate.isDefaultNamespace(root)) {
       // 這里進(jìn)去的是root元素甘凭,一般都會(huì)是默認(rèn)的beans的命名空間
        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 {
       // 存在用戶(hù)完全自定義的配置,使用自定義解析
        delegate.parseCustomElement(root);
    }
}

那么問(wèn)題來(lái)了火邓,delegate.isDefaultNamespace(ele)是如何判斷當(dāng)前元素是否屬于默認(rèn)命名空間的呢丹弱?

public static final String BEANS_NAMESPACE_URI = "http://www.springframework.org/schema/beans";

public boolean isDefaultNamespace(String namespaceUri) {
    return (!StringUtils.hasLength(namespaceUri) || BEANS_NAMESPACE_URI.equals(namespaceUri));
}

public boolean isDefaultNamespace(Node node) {
    return isDefaultNamespace(getNamespaceURI(node));
}   

在xml中以<bean>標(biāo)識(shí)的肯定被系統(tǒng)默認(rèn)為命名空間,如果是類(lèi)似于<context>呢,只能是使用自定義的命名空間去解析了

image.png

如圖圈住的地方铲咨,生成了具體解析的上下文XmlReaderContext躲胳。

XmlBeanDefinitionReader 文件

public XmlReaderContext createReaderContext(Resource resource) {
    return new XmlReaderContext(resource, this.problemReporter, this.eventListener,
            this.sourceExtractor, this, getNamespaceHandlerResolver());
            // 生成具體的XmlReaderContext對(duì)象
}

public NamespaceHandlerResolver getNamespaceHandlerResolver() {
    if (this.namespaceHandlerResolver == null) {
        this.namespaceHandlerResolver = createDefaultNamespaceHandlerResolver();
    }
    return this.namespaceHandlerResolver;
    // 生成了具體的namespace處理器
}

protected NamespaceHandlerResolver createDefaultNamespaceHandlerResolver() {
    return new DefaultNamespaceHandlerResolver(getResourceLoader().getClassLoader());
}

解決了命名空間的問(wèn)題,開(kāi)始具體解析了

public BeanDefinition parseCustomElement(Element ele, 
            BeanDefinition containingBd) {
    String namespaceUri = getNamespaceURI(ele);
    NamespaceHandler handler = this.readerContext.getNamespaceHandlerResolver().resolve(namespaceUri);
    // 這里的handlerResolver就是上面所說(shuō)的命名空間處理器DefaultNamespaceHandlerResolver
    // 然后選擇出具體的handler解析器去解析
    if (handler == null) {
       // 如果沒(méi)有具體的解析器纤勒,那肯定就有問(wèn)題坯苹,需要報(bào)錯(cuò)了
        error("Unable to locate Spring NamespaceHandler for XML schema namespace [" + namespaceUri + "]", ele);
        return null;
    }
    return handler.parse(ele, 
           new ParserContext(this.readerContext, this, containingBd));
}

上面說(shuō)的resolve方法的細(xì)節(jié)在DefaultNamespaceHandlerResolver 類(lèi)

public NamespaceHandler resolve(String namespaceUri) {
    Map<String, Object> handlerMappings = getHandlerMappings();
    // 得到最新的命名空間的情況
    Object handlerOrClassName = handlerMappings.get(namespaceUri);
    if (handlerOrClassName == null) {
        return null;
    }
    else if (handlerOrClassName instanceof NamespaceHandler) {
        return (NamespaceHandler) handlerOrClassName;
    }
    else {
       // 初次生成的map都是<String, String>結(jié)構(gòu)的,需要替換value為NamespaceHandler類(lèi)
        String className = (String) handlerOrClassName;
        try {
            Class<?> handlerClass = ClassUtils.forName(className, this.classLoader);
            // value指明的類(lèi)實(shí)例化
            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.init();
            // 這個(gè)init函數(shù)很重要R√臁4馀取恐仑!后面說(shuō)
            handlerMappings.put(namespaceUri, namespaceHandler);
            return namespaceHandler;
            // 完成了整個(gè)對(duì)需要的命名空間的解析
        }
        catch (ClassNotFoundException ex) {
            throw new FatalBeanException("NamespaceHandler class [" + className + "] for namespace [" +
                    namespaceUri + "] not found", ex);
        }
        catch (LinkageError err) {
            throw new FatalBeanException("Invalid NamespaceHandler class [" + className + "] for namespace [" +
                    namespaceUri + "]: problem with handler class file or dependent class", err);
        }
    }
}

// 延遲加載獲取具體的命名空間詳情信息,生成一個(gè)map結(jié)構(gòu)是<String, String>
private Map<String, Object> getHandlerMappings() {
    if (this.handlerMappings == null) {
        synchronized (this) {
            if (this.handlerMappings == null) {
                try {
                    Properties mappings =
                            PropertiesLoaderUtils.loadAllProperties(this.handlerMappingsLocation, this.classLoader);
                    if (logger.isDebugEnabled()) {
                        logger.debug("Loaded NamespaceHandler mappings: " + mappings);
                    }
                    Map<String, Object> handlerMappings = new ConcurrentHashMap<String, Object>(mappings.size());
                    CollectionUtils.mergePropertiesIntoMap(mappings, handlerMappings);
                    this.handlerMappings = handlerMappings;
                }
                catch (IOException ex) {
                    throw new IllegalStateException(
                            "Unable to load NamespaceHandler mappings from location [" + this.handlerMappingsLocation + "]", ex);
                }
            }
        }
    }
    return this.handlerMappings;
}

如下圖再芋,可以確認(rèn)初始化獲取的map的鍵值對(duì)具體情況


image.png

這是dubbo的命名空間配置


image.png

ContextNamespaceHandler 文件 初始化init注冊(cè)進(jìn)去的解析器

public class ContextNamespaceHandler extends NamespaceHandlerSupport {

    @Override
    public void init() {
        registerBeanDefinitionParser("property-placeholder", new PropertyPlaceholderBeanDefinitionParser());
        registerBeanDefinitionParser("property-override", new PropertyOverrideBeanDefinitionParser());
        registerBeanDefinitionParser("annotation-config", new AnnotationConfigBeanDefinitionParser());
        registerBeanDefinitionParser("component-scan", new ComponentScanBeanDefinitionParser());
        registerBeanDefinitionParser("load-time-weaver", new LoadTimeWeaverBeanDefinitionParser());
        registerBeanDefinitionParser("spring-configured", new SpringConfiguredBeanDefinitionParser());
        registerBeanDefinitionParser("mbean-export", new MBeanExportBeanDefinitionParser());
        registerBeanDefinitionParser("mbean-server", new MBeanServerBeanDefinitionParser());
    }

}

同理菊霜,也可以看看dubbo的

public class DubboNamespaceHandler extends NamespaceHandlerSupport {

    static {
        Version.checkDuplicate(DubboNamespaceHandler.class);
        // 檢查是否重復(fù)了
    }

    public void init() {
        registerBeanDefinitionParser("application", new DubboBeanDefinitionParser(ApplicationConfig.class, true));
        // 如代碼所示,關(guān)鍵字是application,也就是我們配置的**dubbo:application** 字段
        registerBeanDefinitionParser("module", new DubboBeanDefinitionParser(ModuleConfig.class, true));
        registerBeanDefinitionParser("registry", new DubboBeanDefinitionParser(RegistryConfig.class, true));
        registerBeanDefinitionParser("monitor", new DubboBeanDefinitionParser(MonitorConfig.class, true));
        registerBeanDefinitionParser("provider", new DubboBeanDefinitionParser(ProviderConfig.class, true));
        registerBeanDefinitionParser("consumer", new DubboBeanDefinitionParser(ConsumerConfig.class, true));
        registerBeanDefinitionParser("protocol", new DubboBeanDefinitionParser(ProtocolConfig.class, true));
        registerBeanDefinitionParser("service", new DubboBeanDefinitionParser(ServiceBean.class, true));
        registerBeanDefinitionParser("reference", new DubboBeanDefinitionParser(ReferenceBean.class, false));
        registerBeanDefinitionParser("annotation", new DubboBeanDefinitionParser(AnnotationBean.class, true));
    }
}

到這里我們基本清楚了選擇的合適的命名空間去解析數(shù)據(jù)了济赎,可是上文知道一個(gè)命名空間處理器handler有多個(gè)具體的parse解析器鉴逞,那就肯定得選擇一個(gè)具體的解析器去解析∷狙担可看抽象類(lèi) NamespaceHandlerSupport

NamespaceHandlerSupport 文件


public BeanDefinition parse(Element element, ParserContext parserContext) {
    return findParserForElement(element, parserContext).parse(element, parserContext);
    // 返回一個(gè)解析器之后构捡,進(jìn)行parse(element, parserContext)
}

// 果然不出我們所料(當(dāng)然我也是看了源碼的),確實(shí)存在了findPrase這個(gè)方法
private BeanDefinitionParser findParserForElement(Element element, ParserContext parserContext) {
    String localName = parserContext.getDelegate().getLocalName(element);
    
    BeanDefinitionParser parser = this.parsers.get(localName);
    // 通過(guò)元素的名稱(chēng)壳猜,獲取到具體的parse勾徽,例如上面的PropertyPlaceholderBeanDefinitionParser parse解析器
    if (parser == null) {
        parserContext.getReaderContext().fatal(
                "Cannot locate BeanDefinitionParser for element [" + localName + "]", element);
    }
    return parser;
}

可以看看dubbo的解析器具體工作, DubboBeanDefinitionParser 文件

// 解析element元素的具體屬性,返回一個(gè)beandefinition
private static BeanDefinition parse(Element element, ParserContext parserContext,   Class<?> beanClass, boolean required) {
   RootBeanDefinition beanDefinition = new RootBeanDefinition();
   beanDefinition.setBeanClass(beanClass);
   beanDefinition.setLazyInit(false);
   String id = element.getAttribute("id");
   if ((id == null || id.length() == 0) && required) {
    String generatedBeanName = element.getAttribute("name");
    if (generatedBeanName == null || generatedBeanName.length() == 0) {
        if (ProtocolConfig.class.equals(beanClass)) {
            generatedBeanName = "dubbo";
        } else {
            generatedBeanName = element.getAttribute("interface");
        }
   ......

到這里整個(gè)的解析過(guò)程就全部完成了统扳,本篇學(xué)習(xí)了解了生成beandefinition的全過(guò)程喘帚,還有個(gè)問(wèn)題就是spring本身如何持有這些已經(jīng)解析好的beandefinition呢?(PS:是存儲(chǔ)在一個(gè)map容器中)

又得回到我們上面的那幅圖中虛線(xiàn)指出的位置咒钟,在AbstractXmlApplicationContext中就已經(jīng)生成了一個(gè)XmlBeanDefinitionReader對(duì)象了吹由,然后在XmlBeanDefinitionReader中又把自身當(dāng)做一個(gè)參數(shù)傳遞到XmlReaderContext中,而XmlReaderContext又被當(dāng)做一個(gè)參數(shù)傳遞給具體的parse朱嘴。就這樣整個(gè)的鏈路就打通了倾鲫,層層包裝繼承

    public XmlReaderContext createReaderContext(Resource resource) {
        return new XmlReaderContext(resource, this.problemReporter, this.eventListener,
            // 就這個(gè)this!F兼摇N谖簟!很關(guān)鍵的一步
                this.sourceExtractor, this, getNamespaceHandlerResolver());
    }
  • 在系統(tǒng)默認(rèn)的命名空間獲取的方式
protected void processBeanDefinition(Element ele, BeanDefinitionParserDelegate delegate) {
    BeanDefinitionHolder bdHolder = delegate.parseBeanDefinitionElement(ele);
    // 已經(jīng)生成的beandefinition
    if (bdHolder != null) {
        bdHolder = delegate.decorateBeanDefinitionIfRequired(ele, bdHolder);
        try {
            // 最關(guān)鍵的一步H雷贰?牡馈!存儲(chǔ)行冰,getReaderContext() 返回的是XmlReaderContext上下文
            // getRegistry() 獲取其注冊(cè)倉(cāng)庫(kù)捅厂,也就是常說(shuō)的`DefaultListableBeanFactory`
            // 最關(guān)鍵的那個(gè)管理bean的工廠(chǎng)
            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));
    }
}

public static void registerBeanDefinition(
        BeanDefinitionHolder definitionHolder, BeanDefinitionRegistry registry)
        throws BeanDefinitionStoreException {
    String beanName = definitionHolder.getBeanName();
    registry.registerBeanDefinition(beanName, definitionHolder.getBeanDefinition());
    // 這一步就是一直在找的存儲(chǔ)到容器中的步驟

    // Register aliases for bean name, if any.
    String[] aliases = definitionHolder.getAliases();
    if (aliases != null) {
        for (String alias : aliases) {
            registry.registerAlias(beanName, alias);
            // bean配置存在別名的情況,也一并考慮
        }
    }
}
  • 在自定義命名空間獲取的方式

這個(gè)就比較簡(jiǎn)單了资柔,需要對(duì)外提供接口的,持有一個(gè)ParserContext類(lèi)


image.png

就這樣通個(gè)這個(gè)ParserContext類(lèi)把默認(rèn)的BeanDefinitionParserDelegate本身以及xmlreadcontext上下文存儲(chǔ)了撵割,然后傳遞給用戶(hù)自定義的parse進(jìn)行存儲(chǔ)操作贿堰。
當(dāng)然這個(gè)BeanDefinitionParserDelegate是不是用就看用戶(hù)自身了,看了下dubbo是并沒(méi)有使用

所有的過(guò)程都結(jié)束了啡彬,現(xiàn)在spring容器持有beandefinition類(lèi)型的所有bean羹与,還有g(shù)etBean等著我們呢故硅!

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市纵搁,隨后出現(xiàn)的幾起案子吃衅,更是在濱河造成了極大的恐慌,老刑警劉巖腾誉,帶你破解...
    沈念sama閱讀 218,036評(píng)論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件徘层,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡利职,警方通過(guò)查閱死者的電腦和手機(jī)趣效,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,046評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門(mén),熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)猪贪,“玉大人跷敬,你說(shuō)我怎么就攤上這事∪妊海” “怎么了西傀?”我有些...
    開(kāi)封第一講書(shū)人閱讀 164,411評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵,是天一觀(guān)的道長(zhǎng)桶癣。 經(jīng)常有香客問(wèn)我拥褂,道長(zhǎng),這世上最難降的妖魔是什么鬼廓? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,622評(píng)論 1 293
  • 正文 為了忘掉前任肿仑,我火速辦了婚禮,結(jié)果婚禮上碎税,老公的妹妹穿的比我還像新娘尤慰。我一直安慰自己,他們只是感情好雷蹂,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,661評(píng)論 6 392
  • 文/花漫 我一把揭開(kāi)白布伟端。 她就那樣靜靜地躺著,像睡著了一般匪煌。 火紅的嫁衣襯著肌膚如雪责蝠。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 51,521評(píng)論 1 304
  • 那天萎庭,我揣著相機(jī)與錄音霜医,去河邊找鬼。 笑死驳规,一個(gè)胖子當(dāng)著我的面吹牛肴敛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 40,288評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼医男,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼砸狞!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起镀梭,我...
    開(kāi)封第一講書(shū)人閱讀 39,200評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤刀森,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后报账,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體研底,經(jīng)...
    沈念sama閱讀 45,644評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,837評(píng)論 3 336
  • 正文 我和宋清朗相戀三年笙什,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了飘哨。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,953評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡琐凭,死狀恐怖芽隆,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情统屈,我是刑警寧澤胚吁,帶...
    沈念sama閱讀 35,673評(píng)論 5 346
  • 正文 年R本政府宣布,位于F島的核電站愁憔,受9級(jí)特大地震影響腕扶,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜吨掌,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,281評(píng)論 3 329
  • 文/蒙蒙 一半抱、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧膜宋,春花似錦窿侈、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,889評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至肛著,卻和暖如春圆兵,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背枢贿。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,011評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工殉农, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人局荚。 一個(gè)月前我還...
    沈念sama閱讀 48,119評(píng)論 3 370
  • 正文 我出身青樓统抬,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子聪建,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,901評(píng)論 2 355

推薦閱讀更多精彩內(nèi)容

  • Spring Cloud為開(kāi)發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見(jiàn)模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn)茫陆,斷路器金麸,智...
    卡卡羅2017閱讀 134,657評(píng)論 18 139
  • Spring Boot 參考指南 介紹 轉(zhuǎn)載自:https://www.gitbook.com/book/qbgb...
    毛宇鵬閱讀 46,811評(píng)論 6 342
  • 文章作者:Tyan博客:noahsnail.com 3.4 Dependencies A typical ente...
    SnailTyan閱讀 4,159評(píng)論 2 7
  • 愛(ài)美我相信是每個(gè)女生的天性挥下,當(dāng)你每天帶著一樣的妝容,穿著一種衣服的風(fēng)格桨醋,偶爾會(huì)想改變一下自己 其實(shí)這很簡(jiǎn)單棚瘟,用一些...
    枕邊音樂(lè)哦閱讀 574評(píng)論 1 6
  • 會(huì)有人知道你怕黑緊緊抱住你偎蘸,會(huì)有人帶你看遍每一處風(fēng)景,會(huì)有人在你喝醉的時(shí)候哄著你瞬内,會(huì)有人帶你吃遍滿(mǎn)街的美食迷雪,會(huì)有人...
    安梳顏閱讀 9,219評(píng)論 80 245