在博客《一步一步带你深入源码看Spring是如何加载XML配置文件的》中把Spring对XML配置文件如何加载的说明白了,XML配置文件加载完成后就是对标签的解析,本篇博客就是针对Spring bean 标签的解析以及bean definition 的注册。
Spring 中的标签包括默认标签和自定义标签两种, 默认标签的解析是在DefaultBeanDefinitionDocumentReader.parseDefaultElement()方法中进行的, 此方法对4种不同的标签(import,alias,bean和beans)做了不同的处理,具体代码如下:
processBeanDefinition()方法是对默认的bean标签进行解析,具体代码如下:
解析逻辑是:
- 委托BeanDefinitionParserDelegate.parseBeanDefinitionElement()方法进行元素解析,返回BeanDefinitionHolder类型的实例bdHolder,bdHolder实例包含了配置文件中配置的各种属性,例如class,name,id,alias等属性.
- 当返回的bdHolder不为空的情况下若存在默认标签的子节点下还有自定义属性,则还需再次对自定义标签进行解析.
- 解析完成后,需要对解析后的bdHolder进行注册,注册操作委托给了BeanDefinitionReaderUtils.registerBeanDefinition()方法.
- 最后发出响应时间,通知相关的监视器,到此bean就已经加载完成.
parseBeanDefinitionElement()方法调用了重载方法对Spring bean标签进行了犹如剥洋葱似的层层解析.具体代码如下:
从代码中可以看出,解析流程分为4步:
第一步:先提取元素中的id以及name属性.
第二步: 进一步解析其他所有属性并统一封装到AbstractBeanDefinition类型的实例中.具体代码如下:
第三步: 如果检测到bean么有指定beanName,那么使用默认规则为此bean生产beanName;
第四步: 将获取到的信息封装到baanDefinitionHolder的实例中.
对各个标签的解析再次就不一样赘述了,感兴趣的可以自行查看源码.
当spring 中的bean 使用的是默认的标签配置,但是其中的子元素却使用了自定义的标签,例如
<bean id="test" class="test MyClass" >
<mybean:user username="aaa"/>
</bean>
在这个是就是用DefaultBeanDefinitionDocumentReader.parseDefaultElement()方法中decorateBeanDefinitionIfRequired()方法解析默认标签中的自定义标签,具体代码如下:
decorateBeanDefinitionIfRequired()方法调用了重载方法,对元素的所有属性以及子节点进行遍历,代码如下:
从方法中可以看到decorateBeanDefinitionIfRequired()方法还调用了decorateIfRequired()方法获取属性或者元素的命名空间,以此来判断该元素或者属性是否使用于自定义标签的解析条件,找出自定义类型所对应的NamespaceHandler并进行进一步判断,具体代码如下:
到目前为止,配置文件已经完成了解析和装饰,接下来就需要对解析完成的Bean 进行注册,DefaultBeanDefinitionDocumentReader.parseDefaultElement()方法调用了BeanDefinitionReaderUtils.registerBeanDefinition()方法进行注册:
解析的bean 都会被注册到BeanDefinitionRegistry中,bean的注册分为两部分:通过beanName注册以及通过别名进行注册,代码如下:
通过beanName的方式注册bean的实现是DefaultListableBeanFactory.registerBeanDefinition(),主要是有以下四个步骤:
步骤一:对AbstractBeanDefinition的methodOverrides属性进行校验.
步骤二: 对beanName 已经注册的情况进行处理,如果设置了不允许bean的覆盖,则需要抛出异常,否则直接覆盖.
步骤三:加入map缓存
步骤四: 清除之前留下的对应的beanName的缓存.具体代码如下:
通过别名注册bean方法调用的是SimpleAliasRegistry.registerAlias()方法,具体的步骤如下,
步骤一: alias与beanName 相同的情况则不需要处理并删除掉原有alias
步骤二: alias覆盖处理, 若alias 已经使用并已经指向了了另一个beanName则需要用户设置进行处理.
步骤三:alias 循环检查,当A->B存在时,若再出现A->C->B就会抛出异常.
步骤四: 注册alisa;具体代码如下:
到目前为止,bean标签的解析和注册已完成, 当需要对注册bean事件进行监听时可通过注册监听器的方式将处理逻辑写入监听器中.但是Spring没有对时间做任何逻辑处理.