前面四篇文章都是分析 Bean 默认标签的解析过程,包括 基本属性、六个子元素(meta
、lookup-method
、replaced-method
、constructor-arg
、property
、qualifier
),涉及内容较多,拆分成了四篇文章,导致我们已经忘记从哪里出发的了。所以,我们先来回顾下。
DefaultBeanDefinitionDocumentReader 的 #processBeanDefinition(Element ele, BeanDefinitionParserDelegate delegate)
方法,负责 <bean>
标签的解析:
- 在解析过程中,首先调用
BeanDefinitionParserDelegate#parseBeanDefinitionElement(Element ele)
方法,完成默认标签的解析。 - 如果解析成功(返回的
bdHolder != null
),则调用BeanDefinitionParserDelegate#decorateBeanDefinitionIfRequired(Element ele, BeanDefinitionHolder definitionHolder)
方法,完成自定义标签元素的解析。
1. decorateBeanDefinitionIfRequired
前面四篇文章已经分析了默认标签的解析,所以这篇文章分析自定义标签的解析。代码如下:
|
<1>
和<2>
处,都是遍历,前者遍历的是属性(attributes
),后者遍历的是子节点(childNodes
),最终调用的都是#decorateIfRequired(Node node, BeanDefinitionHolder originalDef, BeanDefinition containingBd)
方法,装饰对应的节点( Node )。详细解析,见 「2. decorateIfRequired」 。
2. decorateIfRequired
#decorateIfRequired(Node node, BeanDefinitionHolder originalDef, BeanDefinition containingBd)
方法,装饰对应的节点( Node )。代码如下:
|
- 在
<1>
处,首先获取自定义标签的命名空间。 - 在
<2>
处,如果不是默认的命名空间,则根据该命名空间获取相应的处理器。 - 在
<3>
处,如果处理器存在,则进行装饰处理。
上述过程的详细解析,见 《【死磕 Spring】—— IoC 之解析 标签:解析自定义标签》 一文。
3. 小结
至此,BeanDefinition 的解析过程已经全部完成了,下面做一个简要的总结:
解析 BeanDefinition 的入口在 DefaultBeanDefinitionDocumentReader 的#parseBeanDefinitions(Element root, BeanDefinitionParserDelegate delegate)
方法。该方法会根据命令空间来判断标签是默认标签还是自定义标签,其中:
- 默认标签,由
#parseDefaultElement(Element ele, BeanDefinitionParserDelegate delegate)
方法来实现 - 自定义标签,由 BeanDefinitionParserDelegate 的
#parseCustomElement(Element ele, @Nullable BeanDefinition containingBd)
方法来实现。
在默认标签解析中,会根据标签名称的不同进行 import
、alias
、bean
、beans
四大标签进行处理。其中 bean
标签的解析为核心,它由 processBeanDefinition(Element ele, BeanDefinitionParserDelegate delegate)
方法实现。
processBeanDefinition(Element ele, BeanDefinitionParserDelegate delegate)
方法,开始进入解析核心工作,分为三步:
- 解析默认标签的默认标签:
BeanDefinitionParserDelegate#parseBeanDefinitionElement(Element ele, ...)
方法。该方法会依次解析<bean>
标签的属性、各个子元素,解析完成后返回一个 GenericBeanDefinition 实例对象。 - 解析默认标签下的自定义标签:
BeanDefinitionParserDelegate#decorateBeanDefinitionIfRequired(Element ele, BeanDefinitionHolder definitionHolder)
方法。 - 注册解析的 BeanDefinition:
BeanDefinitionReaderUtils#registerBeanDefinition(BeanDefinitionHolder definitionHolder, BeanDefinitionRegistry registry)
方法。