【死磕 Spring】----- IOC 之解析 bean 标签:解析自定义标签

前面四篇文章都是分析 Bean 默认标签的解析过程,包括基本属性、六个子元素(meta、lookup-method、replaced-method、constructor-arg、property、qualifier),涉及内容较多,拆分成了四篇文章,导致我们已经忘记从哪里出发的了,勿忘初心

processBeanDefinition() 负责 Bean 标签的解析,在解析过程中首先调用BeanDefinitionParserDelegate.parseBeanDefinitionElement() 完成默认标签的解析,如果解析成功(返回的 bdHolder != null ),则首先调用 BeanDefinitionParserDelegate.decorateBeanDefinitionIfRequired() 完成自定义标签元素解析,前面四篇文章已经分析了默认标签的解析,所以这篇文章分析自定义标签的解析。

    public BeanDefinitionHolder decorateBeanDefinitionIfRequired(Element ele, BeanDefinitionHolder definitionHolder) {        return decorateBeanDefinitionIfRequired(ele, definitionHolder, null);    }

调用 decorateBeanDefinitionIfRequired()

    public BeanDefinitionHolder decorateBeanDefinitionIfRequired(            Element ele, BeanDefinitionHolder definitionHolder, @Nullable BeanDefinition containingBd) {        BeanDefinitionHolder finalDefinition = definitionHolder;        // 遍历节点,查看是否有适用于装饰的属性        NamedNodeMap attributes = ele.getAttributes();        for (int i = 0; i < attributes.getLength(); i++) {            Node node = attributes.item(i);            finalDefinition = decorateIfRequired(node, finalDefinition, containingBd);        }        // 遍历子节点,查看是否有适用于修饰的子元素        NodeList children = ele.getChildNodes();        for (int i = 0; i < children.getLength(); i++) {            Node node = children.item(i);            if (node.getNodeType() == Node.ELEMENT_NODE) {                finalDefinition = decorateIfRequired(node, finalDefinition, containingBd);            }        }        return finalDefinition;    }

遍历节点(子节点),调用 decorateIfRequired() 装饰节点(子节点)。

    public BeanDefinitionHolder decorateIfRequired(            Node node, BeanDefinitionHolder originalDef, @Nullable BeanDefinition containingBd) {        // 获取自定义标签的命名空间        String namespaceUri = getNamespaceURI(node);        // 过滤掉默认命名标签        if (namespaceUri != null && !isDefaultNamespace(namespaceUri)) {            // 获取相应的处理器            NamespaceHandler handler = this.readerContext.getNamespaceHandlerResolver().resolve(namespaceUri);            if (handler != null) {                // 进行装饰处理                BeanDefinitionHolder decorated =                        handler.decorate(node, originalDef, new ParserContext(this.readerContext, this, containingBd));                if (decorated != null) {                    return decorated;                }            }            else if (namespaceUri.startsWith("http://www.springframework.org/")) {                error("Unable to locate Spring NamespaceHandler for XML schema namespace [" + namespaceUri + "]", node);            }            else {                if (logger.isDebugEnabled()) {                    logger.debug("No Spring NamespaceHandler found for XML schema namespace [" + namespaceUri + "]");                }            }        }        return originalDef;    }

首先获取自定义标签的命名空间,如果不是默认的命名空间则根据该命名空间获取相应的处理器,最后调用处理器的 decorate() 进行装饰处理。具体的装饰过程这里不进行讲述,在后面分析自定义标签时会做详细说明。

至此,Bean 的解析过程已经全部完成了,下面做一个简要的总结。

解析 BeanDefinition 的入口在 DefaultBeanDefinitionDocumentReader.parseBeanDefinitions() 。该方法会根据命令空间来判断标签是默认标签还是自定义标签,其中默认标签由 parseDefaultElement() 实现,自定义标签由 parseCustomElement() 实现。在默认标签解析中,会根据标签名称的不同进行 import 、alias 、bean 、beans 四大标签进行处理,其中 bean 标签的解析为核心,它由 processBeanDefinition() 方法实现。 processBeanDefinition() 开始进入解析核心工作,分为三步:

  1. 解析默认标签: BeanDefinitionParserDelegate.parseBeanDefinitionElement()
  2. 解析默认标签下的自定义标签: BeanDefinitionParserDelegate.decorateBeanDefinitionIfRequired()
  3. 注册解析的 BeanDefinition: BeanDefinitionReaderUtils.registerBeanDefinition

在默认标签解析过程中,核心工作由 parseBeanDefinitionElement() 方法实现,该方法会依次解析 Bean 标签的属性、各个子元素,解析完成后返回一个 GenericBeanDefinition 实例对象。

原文发布于微信公众号 - Java技术驿站(chenssy89)

原文发表时间:2018-09-26

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏腾讯Bugly的专栏

深入浅出 Retrofit,这么牛逼的框架你们还不来看看?

Android 开发中,从原生的 HttpUrlConnection 到经典的 Apache 的 HttpClient,再到对前面这些网络基础框架的封装,比如 ...

38960
来自专栏Android开发指南

Android优化指南

52870
来自专栏后端沉思录

Spring源码之解析并注册BeanDefinition(一)

最近有空把Spring加载bean流程复习了一下,也乘机可以做个整理.首先还是看下入口代码,本文主要讲解析及注册BeanDefinition整体加载流程:

21210
来自专栏向治洪

系统捕获异常并发送到服务器

大家都知道,现在安装Android系统的手机版本和设备千差万别,在模拟器上运行良好的程序安装到某款手机上说不定就出现崩溃的现象,开发者个人不可能购买所有设备逐个...

21070
来自专栏葡萄城控件技术团队

扩展GridView控件——为内容项添加拖放及分组功能

引言 相信大家对GridView都不陌生,是非常有用的控件,用于平铺有序的显示多个内容项。打开任何WinRT应用或者是微软合作商的网站,都会在APP中发现Gri...

37050
来自专栏java 成神之路

spring 之 import标签、alias标签、beans标签 解析

604100
来自专栏JavaEdge

IoC容器的初始化过程(上)1 BeanDefinition的Resource定位

31870
来自专栏程序员叨叨叨

【转-干货】Retrofit2.0使用总结及注意事项

随着Google对HttpClient 摒弃,和Volley的逐渐没落,OkHttp开始异军突起,而Retrofit则对okHttp进行了强制依赖。Retrof...

99320
来自专栏我和未来有约会

如何在silverlihgt中使用右键

一般我们在silverlight中点击右键会出现如下的对话筐. ? ? 在flash中 其提供了一个可定制话的右键菜单系统.(ContextMenu) 这个...

21570
来自专栏Golang语言社区

GoLang并发控制(下)

context的字面意思是上下文,是一个比较抽象的词,字面上理解就是上下层的传递,上会把内容传递给下,在go中程序单位一般为goroutine,这里的上下文便是...

33830

扫码关注云+社区

领取腾讯云代金券