Spring的XML解析原理,这一次全搞懂再走( 七 )
public BeanDefinitionHolder decorateBeanDefinitionIfRequired(Element ele, BeanDefinitionHolder definitionHolder, @Nullable BeanDefinition containingBd) {BeanDefinitionHolder finalDefinition = definitionHolder;//根据bean标签属性装饰BeanDefinitionHolder , 比如NamedNodeMap attributes = ele.getAttributes();for (int i = 0; i < attributes.getLength(); i++) {Node node = attributes.item(i);finalDefinition = decorateIfRequired(node, finalDefinition, containingBd);}//根据bean标签子元素装饰BeanDefinitionHolder\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; }
在这个方法中分别对Bean标签的属性和子标签迭代 , 获取其中的自定义标签进行解析 , 并装饰之前创建的BeanDefinition对象 , 如同下面的c和p:
// c:和p:表示通过构造器和属性的setter方法给属性赋值 , 是constructor-arg和property的简化写法
两个步骤是一样的 , 我们点进decorateIfRequired方法中:
public BeanDefinitionHolder decorateIfRequired(Node node, BeanDefinitionHolder originalDef, @Nullable BeanDefinition containingBd) {//根据node获取到node的命名空间 , 形如:String namespaceUri = getNamespaceURI(node);if (namespaceUri != nullif (handler != null) {//调用NamespaceHandler处理类的decorate方法 , 开始具体装饰过程 , 并返回装饰完的对象BeanDefinitionHolder decorated =handler.decorate(node, originalDef, new ParserContext(this.readerContext, this, containingBd));if (decorated != null) {return decorated;}}else if (namespaceUri.startsWith("")) {error("Unable to locate Spring NamespaceHandler for XML schema namespace [" + namespaceUri + "]", node);}else {// A custom namespace, not to be handled by Spring - maybe "xml:...".if (logger.isDebugEnabled()) {logger.debug("No Spring NamespaceHandler found for XML schema namespace [" + namespaceUri + "]");}}}return originalDef; }
这里也和我们之前说的一样 , 首先获取到标签对应的namespaceUri , 然后通过这个Uri去获取到对应的NamespceHandler , 最后再调用NamespceHandler的decorate方法进行装饰 。 我们先来看看获取NamespceHandler的过程 , 这涉及到一个非常重要的高扩展性的思想——SPI(有关SPI , 在我之前的文章Dubbo——SPI及自适应扩展原理中已经详细讲解过 , 这里不再赘述):
public NamespaceHandler resolve(String namespaceUri) {// 获取spring中所有jar包里面的 "META-INF/spring.handlers"文件 , 并且建立映射关系Map handlerMappings = getHandlerMappings();//根据namespaceUri: , 获取到这个命名空间的处理类Object handlerOrClassName = handlerMappings.get(namespaceUri);if (handlerOrClassName == null) {return null;}else if (handlerOrClassName instanceof NamespaceHandler) {return (NamespaceHandler) handlerOrClassName;}else {String className = (String) handlerOrClassName;try {Class> handlerClass = ClassUtils.forName(className, this.classLoader);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);//调用处理类的init方法 , 在init方法中完成标签元素解析类的注册namespaceHandler.init();handlerMappings.put(namespaceUri, namespaceHandler);return namespaceHandler;}} }// AOP标签对应的NamespaceHandler , 可以发现NamespaceHandler的作用就是管理和注册与自己相关的标签解析器 public void init() {// In 2.0 XSD as well as in 2.1 XSD.registerBeanDefinitionParser("config", new ConfigBeanDefinitionParser());registerBeanDefinitionParser("aspectj-autoproxy", new AspectJAutoProxyBeanDefinitionParser());registerBeanDefinitionDecorator("scoped-proxy", new ScopedProxyBeanDefinitionDecorator());// Only in 2.0 XSD: moved to context namespace as of 2.1registerBeanDefinitionParser("spring-configured", new SpringConfiguredBeanDefinitionParser()); }
- 多就是好吗?解析智能手机多摄发展困局
- 新基建下,系统集成商数字化建设及渠道管理深度解析
- 四个维度解析区域发展与治理
- 基于Spring+Angular9+MySQL开发平台
- 智媒视角看深圳用数据解析城市
- “联邦的战斗力量”再现,华硕Z490主板机动战士高达版解析
- 天翼云全国首个工业互联网平台二级解析节点上线
- 在美国当快递小哥赚钱吗?西瓜视频解析除了努力,运气也很重要
- 蚂蚁庄园月有阴晴圆缺答案解析 12月18日今天支付宝蚂蚁庄园答案大全
- 如何做到性能翻倍 NVIDIA Ampere架构解析