Spring源码深度解析(五)加载Bean
之前提到的在XmlBeanFactory构造函数中调用了XmlBeanDefinitionReader类型的reader属性提供的方法this.reader.loadBeanDefinitions(resource),而这句代码则是整个资源加载的切入点,我们先来看看这个方法的时序图,如图2-9所示:
看到图2-9我们才知道,原来饶了这么久还没有切入正题,还一直在为加载XML文件和解析注册Bean在做准备工作。我们根据时序图分析下都咋在准备什么?
- 封装资源文件。当进入XmlBeanDefinitionReader后首先对参数Resource使用EncodeResource类进行封装。
- 获取输入流。从Resource中获取对应的InputStream并构造InputSource。
- 通过构造的InputSource实例和Resource实例继续调用函数doLoadBeanDefinitions.
我们来看一下loadBeanDefinitions函数具体的实现过程:
那么EncodeResour的作用是什么呢?通过名称,我们可以大致推断这个类主要是用于对资源文件的编码进行处理。其中的主要逻辑体现在getReader()方法中,当设置了编码的时候spring会使用相应的编码作为输入流的编码。
当构造好encodeResource对象后,再次转入了可复用方法loadBeanDefinitions(new EncodedResource(resource)).
这个方法内部才是真正的数据准备阶段,也就是时序图锁描述的逻辑:
我们再次准备一下数据准备阶段的逻辑,首先对传入的resource参数做封装,目的是考虑到Resource可能存在编码要求的问题,其次,通过SAX读取XML文件的方式来准备InputSource对象,最后将准备的数据通过参数传入真正的核心处理部分doLoadBeanDefinitions(inputSource,encodedResource.getResource())。
上面的代码只做了三件事,每一件都是必不可少的。
- 获取对XML文件的验证模式。
- 加载XML文件,并得到对应的Document.
- 根据返回的Document注册Bean信息。