Spring源码系列 第二篇 XML解析
- XML解析
- spring.xml两种标签解析方式
- 1.默认标签解析
- 2.自定义标签解析
- 自定义标签解析流程
XML解析
spring.xml两种标签解析方式
1.默认标签解析
默认标签解析是通过流的方式读取spring.xml文件,把文件封装为document对象,再通过预定义的bean import等标签,获取对应的标签值,并最终封装为beanDefinition对象的过程
1.程序入口
ClassPathXmlApplicationContext applicationContext = new ClassPathXmlApplicationContext("spring.xml");
2.refresh方法
refresh();
3.obtainFreshBeanFactory核心方法
此方法最主要两个功能
- 创建BeanFactory对象
- xml解析
ConfigurableListableBeanFactory beanFactory = obtainFreshBeanFactory();
4.refreshBeanFactory方法
refreshBeanFactory()
- hasBeanFactory(), 判断是否有一个BeanFactory对象 如果有 消除beanFactory工厂中的实例对象
- createBeanFactory(),创建BeanFactory实例工厂
- customizeBeanFactory,1.设置是否允许循环依赖设置方法:实现BeanDefinitionRegistryPostProcessor接口,在postProcessBeanFactory方法中设置beanFactory 对象中 allowCircularReferences 为true2.设置是否允许使用相同名称重新注册不同的bean实现
- loadBeanDefinitions,解析xml 并把xml中的标签封装为beanDefinition对象
- doLoadDocument,解析inputSource流 把流对象转换为Document文档对象
- registerBeanDefinitions,核心方法,根据Document对象 拿到xml对象中的标签元素 并封装为BeanDefinition对象
- parseBeanDefinitions(),具体的标签解析流程
- parseDefaultElement(),具体的默认标签解析方法,解析默认标签并封装为beanDefinition对象
2.自定义标签解析
自定义标签前面的流程和默认标签解析的代码流程基本差不多,只在具体解析时有差别,总的来说,默认标签解析是通过流的方式来获取xml中的标签值,自定义标签解析是通过标签头获取对应的xsd的值,再去固定的spring.handlers文件中找到对应的映射关系,再通过paser方法完成标签解析,是一种spi的设计思想
delegate.parseCustomElement(ele);
自定义标签解析流程
1,根据标签头Element对象获取对应的xsd的值就是namespaceUri 即spring.xml中最上方的类似于网址的东西
String namespaceUri = getNamespaceURI(ele);
2.resolve方法 加载spring所有jar中的spring.handlers文件,并建立映射关系
NamespaceHandler handler = this.readerContext.getNamespaceHandlerResolver().resolve(namespaceUri);
3,根据namespaceUri从映射关系中找到对应的实现了NamespaceHandler接口的类
NamespaceHandler namespaceHandler = (NamespaceHandler) BeanUtils.instantiateClass(handlerClass);
4,调用类的init方法,init方法是注册了各种自定义标签的解析类
namespaceHandler.init();
5,根据namespaceUri找到对应的解析类,然后调用paser方法完成标签解析
return handler.parse(ele, new ParserContext(this.readerContext, this, containingBd));