问题
在配置文件中配置<dubbo>这种标签的,那么spring是如何获取到<dubbo>标签里面配置的信息。
解决方式
Spring可扩展Schema提供自定义配置支持,那么Spring扩展Schema的步骤是什么?
- 设计配置属性和JavaBean
- 编写XSD文件
- 编写NamespaceHandler和BeanDefinitionParser完成解析工作
- 编写spring.handlers和spring.schemas串联起所有部件
- 在Bean文件中应用
Dubbo的解析流程
- 在项目的META-INF/spring.handlers下面信息
http\://code.alibabatech.com/schema/dubbo=com.alibaba.dubbo.config.spring.schema.DubboNamespaceHandler
Spring在遇到dubbo名称空间时,会回调DubboNamespaceHandler。
- 加载dubbo的Schemas
http\://code.alibabatech.com/schema/dubbo/dubbo.xsd=META-INF/dubbo.xsd
- DubboNamespaceHandler 集成了Spring的NamespaceHandlerSupport
public class DubboNamespaceHandler extends NamespaceHandlerSupport {
static {
Version.checkDuplicate(DubboNamespaceHandler.class);
}
@Override
public void init() {
// 调用它来注册提供的BeanDefinitionParser来处理指定的元素
registerBeanDefinitionParser("application", new DubboBeanDefinitionParser(ApplicationConfig.class, true));
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 AnnotationBeanDefinitionParser());
}
}
- 所有dubbo的标签,都统一用DubboBeanDefinitionParser进行解析,基于一对一属性映射,将XML标签解析为Bean对象。
-> DefaultBeanDefinitionDocumentReader#registerBeanDefinitions
->doRegisterBeanDefinitions
-> createDelegate
-> BeanDefinitionParserDelegate
-> preProcessXml 处理前 (留给子类实现)
-> parseBeanDefinitions
-> parseDefaultElement(这里面根据Element的name去使用不同的解析)
-> processBeanDefinition 这里主要解析下这个
-> BeanDefinitionParserDelegate解析xml为BeanDefinitionHolder
-> decorateBeanDefinitionIfRequired装饰该bean 依赖的子类的
-> BeanDefinitionReaderUtils.registerBeanDefinition 向DefaultListableBeanFactory注册beanDefinition
将beanDefinition保存到DefaultListableBeanFactory#beanDefinitionMap
-> fireComponentRegistered 发送注册事件
-> BeanDefinitionParserDelegate#parseCustomElement -- 这里是处理继承NamespaceHandler接口的,实际上就是spring自定义标签解析
-> postProcessXml 处理后 (留给子类实现)