Spring源码分析——解析Xml过程

Easter79
• 阅读 422

图1

Spring源码分析——解析Xml过程

上次看到doRegisterBeanDefinitions方法,是DefaultBeanDefinitionDocumentReader这个类,这里是真正开始解析xml。

那么先来看下doRegisterBeanDefinitions做了些什么,先调用了createDelegate方法,如图2:

图2

Spring源码分析——解析Xml过程

实例化了一个BeanDefinitionParserDelegate(这里用了代理模式,不知道对不对),在来看下initDefaults方法,如图3:

图3

Spring源码分析——解析Xml过程

能看到有个populateDefaults方法,defaults全局变量是DocumentDefaultsDefinition defaults = new DocumentDefaultsDefinition();这个类包含了最外层的一些属性,关键还是看下populateDefaults方法,如图4:

图4

Spring源码分析——解析Xml过程

方法太长用两张截图

Spring源码分析——解析Xml过程

这里是解析最外层的的一些属性,一共有6个:

  • default-lazy-init:延迟加载,延迟实例化bean;
  • default-merge:在继承关系时在子类中合并父类的值;
  • default-autowire:按哪种方式注入,byType、byName等;
  • default-autowire-candidates:匹配到的bean才能加载;
  • default-init-method:初始化方法;
  • default-destroy-method:销毁方法;

能看到最后有一个setSource方法,这个可以自己实现SourceExtractor,额外的source信息,通过XmlBeanDefinitionReader.setSourceExtractor(SourceExtractor);来实现。如图5:

图5

Spring源码分析——解析Xml过程

使用的话如图6:

图6

Spring源码分析——解析Xml过程

好了那么继续下看,在解析完这些属性以后,有个this.readerContext.fireDefaultsRegistered(this.defaults);方法,如图7:

图7

Spring源码分析——解析Xml过程

这里有个eventListener,默认是EmptyReaderEventListener,可以自己实现ReaderEventListener以后覆盖掉,具体用法和SourceExtractor是一样的,这里不再累述了。

那创建完代理类就是真正的解析啦,就是parseBeanDefinitions(root, this.delegate);方法,如图8:

图8

Spring源码分析——解析Xml过程

关于自定义标签我们下次再讲,还是来看下parseDefaultElement(ele, delegate);方法把,如图9:

图9

Spring源码分析——解析Xml过程

很明确就是解析四个标签:,先来看下在复杂也最繁琐的解析吧,如图10:

图10

Spring源码分析——解析Xml过程

这个方法就是处理解析并完成注册的方法,代理方法开始显威啦delegate.parseBeanDefinitionElement(ele);如图11:

图11

Spring源码分析——解析Xml过程

能看到先取id和name,默认的bean name是id,但是如果id没有,name就是bean name,在配置中name还可以用",; "分隔开,一个bean可以拥有多个别名,最后别名放到哪里我们后面会看到,如图12:

图12

Spring源码分析——解析Xml过程

刚只是方法的前半部分,还有后半部分,如图13:

图13

Spring源码分析——解析Xml过程

parseBeanDefinitionElement(ele, beanName, containingBean);方法就是解析中的所有属性,如图14:

图14

Spring源码分析——解析Xml过程

这是方法的前半部分,其实没啥好看的,关键还得看后半部分,如图15:

图15

Spring源码分析——解析Xml过程

这里就是解析里面所有的attribute了,具体这里就不在往下挖了因为实在太多了,最后所有解析出来的值都放入了AbstractBeanDefinition类中,好奇的小伙伴肯定想里面到底有多少属性,这里来列一下:

  1. scope
  2. abstract
  3. lazy-init
  4. autowire
  5. depends-on
  6. autowire-candidate
  7. primary
  8. init-method
  9. destroy-method
  10. factory-method
  11. factory-bean

具体怎么使用请看官网。

那么好了bean的所有信息都拿到了,接下去干嘛呢,回过头看图13,如果没有定义beanName的话,只有class,这个时候会去生成一个beanName,类似于com.shenqi.test.abc#0,如果有第二个就是#1这样以此类推。好,那么这个时候就返回了一个BeanDefinitionHolder,回过头来看图10的BeanDefinitionReaderUtils.registerBeanDefinition(bdHolder, getReaderContext().getRegistry());方法,如图16:

图16

Spring源码分析——解析Xml过程

拿到了beanName,最后放入到DefaultListableBeanFactory,BeanDefinition放入Map<String, BeanDefinition> beanDefinitionMap = new ConcurrentHashMap<>(256);,beanName放入private volatile List beanDefinitionNames = new ArrayList<>(256);中,别名的话会放入SimpleAliasRegistry的private final Map<String, String> aliasMap = new ConcurrentHashMap<>(16);中。

那么关于解析的话,其实最后就是放入SimpleAliasRegistry的private final Map<String, String> aliasMap = new ConcurrentHashMap<>(16);中。

其实就是把读取xml和解析xml重新走一遍。

内嵌的的话就是解析图8的parseBeanDefinitions(root, this.delegate);重新走一遍。

这次一些细节的地方没有说明,大家可以自行去研究下,比如做获取,lookup,怎么使用等。

下次的话就是自定义啦。

如有问题请提出,共同提高,谢谢!!

点赞
收藏
评论区
推荐文章
blmius blmius
3年前
MySQL:[Err] 1292 - Incorrect datetime value: ‘0000-00-00 00:00:00‘ for column ‘CREATE_TIME‘ at row 1
文章目录问题用navicat导入数据时,报错:原因这是因为当前的MySQL不支持datetime为0的情况。解决修改sql\mode:sql\mode:SQLMode定义了MySQL应支持的SQL语法、数据校验等,这样可以更容易地在不同的环境中使用MySQL。全局s
Wesley13 Wesley13
3年前
java将前端的json数组字符串转换为列表
记录下在前端通过ajax提交了一个json数组的字符串,在后端如何转换为列表。前端数据转化与请求varcontracts{id:'1',name:'yanggb合同1'},{id:'2',name:'yanggb合同2'},{id:'3',name:'yang
皕杰报表之UUID
​在我们用皕杰报表工具设计填报报表时,如何在新增行里自动增加id呢?能新增整数排序id吗?目前可以在新增行里自动增加id,但只能用uuid函数增加UUID编码,不能新增整数排序id。uuid函数说明:获取一个UUID,可以在填报表中用来创建数据ID语法:uuid()或uuid(sep)参数说明:sep布尔值,生成的uuid中是否包含分隔符'',缺省为
待兔 待兔
4个月前
手写Java HashMap源码
HashMap的使用教程HashMap的使用教程HashMap的使用教程HashMap的使用教程HashMap的使用教程22
Jacquelyn38 Jacquelyn38
3年前
2020年前端实用代码段,为你的工作保驾护航
有空的时候,自己总结了几个代码段,在开发中也经常使用,谢谢。1、使用解构获取json数据let jsonData  id: 1,status: "OK",data: 'a', 'b';let  id, status, data: number   jsonData;console.log(id, status, number )
Wesley13 Wesley13
3年前
mysql设置时区
mysql设置时区mysql\_query("SETtime\_zone'8:00'")ordie('时区设置失败,请联系管理员!');中国在东8区所以加8方法二:selectcount(user\_id)asdevice,CONVERT\_TZ(FROM\_UNIXTIME(reg\_time),'08:00','0
Wesley13 Wesley13
3年前
00:Java简单了解
浅谈Java之概述Java是SUN(StanfordUniversityNetwork),斯坦福大学网络公司)1995年推出的一门高级编程语言。Java是一种面向Internet的编程语言。随着Java技术在web方面的不断成熟,已经成为Web应用程序的首选开发语言。Java是简单易学,完全面向对象,安全可靠,与平台无关的编程语言。
Stella981 Stella981
3年前
Django中Admin中的一些参数配置
设置在列表中显示的字段,id为django模型默认的主键list_display('id','name','sex','profession','email','qq','phone','status','create_time')设置在列表可编辑字段list_editable
Wesley13 Wesley13
3年前
MySQL部分从库上面因为大量的临时表tmp_table造成慢查询
背景描述Time:20190124T00:08:14.70572408:00User@Host:@Id:Schema:sentrymetaLast_errno:0Killed:0Query_time:0.315758Lock_
Python进阶者 Python进阶者
10个月前
Excel中这日期老是出来00:00:00,怎么用Pandas把这个去除
大家好,我是皮皮。一、前言前几天在Python白银交流群【上海新年人】问了一个Pandas数据筛选的问题。问题如下:这日期老是出来00:00:00,怎么把这个去除。二、实现过程后来【论草莓如何成为冻干莓】给了一个思路和代码如下:pd.toexcel之前把这
Easter79
Easter79
Lv1
今生可爱与温柔,每一样都不能少。
文章
2.8k
粉丝
5
获赞
1.2k