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

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

  1. public BeanDefinitionHolder decorateBeanDefinitionIfRequired(Element ele, BeanDefinitionHolder definitionHolder) {

  2. return decorateBeanDefinitionIfRequired(ele, definitionHolder, null);

  3. }

调用 decorateBeanDefinitionIfRequired()

  1. public BeanDefinitionHolder decorateBeanDefinitionIfRequired(

  2. Element ele, BeanDefinitionHolder definitionHolder, @Nullable BeanDefinition containingBd) {

  3. BeanDefinitionHolder finalDefinition = definitionHolder;

  4. // 遍历节点,查看是否有适用于装饰的属性

  5. NamedNodeMap attributes = ele.getAttributes();

  6. for (int i = 0; i < attributes.getLength(); i++) {

  7. Node node = attributes.item(i);

  8. finalDefinition = decorateIfRequired(node, finalDefinition, containingBd);

  9. }

  10. // 遍历子节点,查看是否有适用于修饰的子元素

  11. NodeList children = ele.getChildNodes();

  12. for (int i = 0; i < children.getLength(); i++) {

  13. Node node = children.item(i);

  14. if (node.getNodeType() == Node.ELEMENT_NODE) {

  15. finalDefinition = decorateIfRequired(node, finalDefinition, containingBd);

  16. }

  17. }

  18. return finalDefinition;

  19. }

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

  1. public BeanDefinitionHolder decorateIfRequired(

  2. Node node, BeanDefinitionHolder originalDef, @Nullable BeanDefinition containingBd) {

  3. // 获取自定义标签的命名空间

  4. String namespaceUri = getNamespaceURI(node);

  5. // 过滤掉默认命名标签

  6. if (namespaceUri != null && !isDefaultNamespace(namespaceUri)) {

  7. // 获取相应的处理器

  8. NamespaceHandler handler = this.readerContext.getNamespaceHandlerResolver().resolve(namespaceUri);

  9. if (handler != null) {

  10. // 进行装饰处理

  11. BeanDefinitionHolder decorated =

  12. handler.decorate(node, originalDef, new ParserContext(this.readerContext, this, containingBd));

  13. if (decorated != null) {

  14. return decorated;

  15. }

  16. }

  17. else if (namespaceUri.startsWith("http://www.springframework.org/")) {

  18. error("Unable to locate Spring NamespaceHandler for XML schema namespace [" + namespaceUri + "]", node);

  19. }

  20. else {

  21. if (logger.isDebugEnabled()) {

  22. logger.debug("No Spring NamespaceHandler found for XML schema namespace [" + namespaceUri + "]");

  23. }

  24. }

  25. }

  26. return originalDef;

  27. }

首先获取自定义标签的命名空间,如果不是默认的命名空间则根据该命名空间获取相应的处理器,最后调用处理器的 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 实例对象。

原文发布时间为:2018-09-26
本文作者:Java技术驿站
本文来自云栖社区合作伙伴“Java技术驿站”,了解相关信息可以关注“Java技术驿站”。

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

  1. 死磕Spring系列:SpringAOP源码解析

    1. 术语 在SpringAOP中,有很多专业术语,已经了解的小伙伴可以略过,不了解的朋友就需要注意看一下啦,不然会很懵的哟. Aspect(切面) 跨多个类的关注点的模块化,与垂直调用的业务树不同, ...

  2. 【死磕 Spring】—– 死磕 Spring 精品合集

    Java面经 2019-03-05 09:00:40 原文:http://cmsblogs.com/?cat=206 [死磕 Spring]-– IOC 之深入理解 Spring IoC [死磕 Sp ...

  3. [死磕 Spring 21/43] --- IOC 之 Factory 实例化 bean

    引用原文: https://www.cmsblogs.com/article/1391375394573258752 [死磕 Spring 21/43] - IOC 之 Factory 实例化 bea ...

  4. 【死磕 Spring】----- IOC 之 Factory 实例化 bean

    原文:https://www.cmsblogs.com/category/1391374860344758272 『chenssy』 这篇我们关注创建 bean 过程中的第一个步骤:实例化 bean, ...

  5. 【死磕 Spring】----- IOC 之 加载 Bean

    原文:https://www.cmsblogs.com/category/1391374860344758272 『chenssy』 先看一段熟悉的代码: ClassPathResource reso ...

  6. preg_match 参数获取两个_「死磕 Spring」—– IOC 之 获取 Document 对象

    原文出自:http://cmsblogs.com作者:chenssy 在 XmlBeanDefinitionReader.doLoadDocument() 方法中做了两件事情,一是调用 getVali ...

  7. 死磕Spring AOP系列2:剖析Bean处理器之BeanNameAutoProxyCreator

    通过前一篇<死磕Spring AOP系列1:编程式实现AOP>,学习了Spring对代理的底层支持,认识了ProxyFactory对象,及从类设计层面认识了PointCut&Adv ...

  8. Spring IoC容器与Bean管理

    Spring IoC容器与Bean管理 一.Spring IoC容器与Bean管理 1.Spring快速入门 IoC控制反转 DI依赖注入 Spring概述 Spring IoC初体验 使用XML方式 ...

  9. java基础巩固-宇宙第一AiYWM:为了维持生计,Spring全家桶_Part1-3(学学Spring源码呗:默认的标签和自定义标签是咋解析的)~整起

    Part3:上一次说到了Spring的DefaultBeanDefinitionDocumentReader类中的parseBeanDefinitions(Element root, BeanDefi ...

最新文章

  1. 【深度学习篇】--神经网络中的调优二,防止过拟合
  2. PHP魔术方法__tostring()篇
  3. 四大原因可能让你拿到订单
  4. 后台开发经典书籍--Redis深度历险:核心原理和应用实践
  5. 现代密码学1.2--Kerckhoffs原则
  6. 13、创建触发器(CREATE TRIGGER)
  7. myisam怎么读_耗时半年,我成功“逆袭”,拿下美团offer(刷面试题+读源码+项目准备)...
  8. android intent 跳转卡顿_Intent七大属性
  9. 比利时女摄影师玛瑞亚
  10. 环形buffer缓冲区
  11. 看下资深架构师平时需要解决的问题,对比你离资深架构师还有多少距离——再论技术架构的升级之路...
  12. python集合类型中的元素是有序的_python基础篇:很有意思的数据类型,集合(set) 常用去去重...
  13. mybatis-plus 中 queryWrapper and与or嵌套
  14. “程序员不会数据分析,有什么影响?”资深程序员:基本等于自废武功!
  15. 正则表达式之全部符号解释
  16. 百度搜索关键词纠错机制研究
  17. sd卡无法完成格式化解决办法
  18. 请不要在“微服务”的狂热中迷失自我!
  19. 考试系统设计oracle,在线考试系统的设计与实现|毕业设计源代码|论文开题报告|Oracle...
  20. Js节流(防连点)和防抖动

热门文章

  1. json为全局变量 vue_vue 设置全局变量、指定请求的 baseurl
  2. ux设计中的各种地图_如何在UX设计中使用颜色
  3. 三年经验前端社招——慧择网
  4. nginx修改upstream不重启的方法(ngx_http_dyups_module模块)
  5. server 2008R2 AD域环境中DHCP服务器的授权步骤
  6. Cobbler部署之FAQ处理
  7. mysql语法学习(一)__Instances__表
  8. opentaps mysql_opentaps 1.4 联接 mysql 笔记
  9. logcat崩溃_使用logcat抓取Android崩溃日志
  10. Python档案袋( 面向对象 )