Spring源码之XML文件中Bean标签的解析1

读取XML文件,创建对象

xml文件里包含Bean的信息,为了避免多次IO,需要一次性读取xml文件中所有bean信息,加入到Spring工厂。

读取配置文件

java 复制代码
new ClassPathResource("applicationContext.xml")

ClassPathResource是Spring封装的一个类型;

Resource接口 :可以读取相关资源文件的内容 获得输入流;可读取的类型,不仅包括本地的xml、 properties、txt 等文件,还包括 网络中的资源。它有父接口,InputStreamSource,其中定义了一个getInputStream方法,用来获取输入流。

ClassPathResource类中对getInputStream方法的实现。

Resource将xml配置文件读取到jvm中,那jvm中是如何体现xml中的bean呢?

因为一切皆对象,所以肯定是以对象的方式存在的。标签就会以BeanDefinition对象的方式存在于jvm中。

Spring通过SAX的技术手段,对xml解析,封装BeanDefiniton。

IOC的核心

下方代码所代表的含义,实际上就是IOC的核心:

java 复制代码
BeanFactory beanFactory = new XmlBeanFactory(new ClassPathResource("applicationContext.xml"));

它包括以下几个问题:

  1. 怎么读取配置文件 获得IO资源
  2. 读取配置文件后,如何在Spring中以对象的形式进行封装
  3. 根据配置信息创建对象
  4. 所创建对象的生命周期
怎么读取配置文件 获得IO资源

通过Resource接口的实现类,比如ClassPathResource,得到InputStream。

读取配置文件后,如何在Spring中以对象的形式进行封装

以BeanDefinition的形式存在,用的最多的实现类是GenericBeanDefinition。

Spring底层如何通过读取 XML 封装成BeanDefinition呢

使用XmlBeanFactory中的XmlBeanDefinitionReader对象的方法。

Spring标识XmlBeanFactory过期了,可以用以下代码替换:

java 复制代码
DefaultListableBeanFactory beanFactory1 = new DefaultListableBeanFactory();
Resource resource = new ClassPathResource("applicationContext.xml");
XmlBeanDefinitionReader xmlBeanDefinitionReader = new XmlBeanDefinitionReader(beanFactory1);
xmlBeanDefinitionReader.loadBeanDefinitions(resource);

Object product1 = beanFactory1.getBean("product"); 
System.out.println("product1 = " + product1);

但是比较繁琐,我们分析源码还是使用XmlBeanFactory来分析。

Spring底层如何做到通过XmlBeanDefinitionReader读取 XML封装成BeanDefinition呢?

Spring允许在一个工程中有多个Spring工厂同时出现 ,但是情况非常少见 。

比如SpringMVC中 父子容器

DispatcherServlet ---- childFactory

ContextLoaderListener ---- rootFactory

markdown 复制代码
InputStream inputStream = encodedResource.getResource().getInputStream();
这行代码是封装成SAX的InputStream类型,对xml进行解析。
最主要的就是return doLoadBeanDefinitions(inputSource, encodedResource.getResource());
来加载BeanDefinition。
markdown 复制代码
Document doc = doLoadDocument(inputSource, resource);
这行代码得到的还是XML解析封装的对象。
int count = registerBeanDefinitions(doc, resource);
这行代码就是注册BeanDefinition的方法,并返回注册的数量。
markdown 复制代码
parseDefaultElement(ele, delegate);用来解析基本标签
	比如:
		<bean id="" class="" scope="" parent="" init-method=""
            <property name  value
          </bean>
         <bean id="" class="" scope="" parent="" init-method=""
            <construt-arg>
         </bean>
delegate.parseCustomElement(ele);解析自定义标签
	比如:
	<context:propertyplace-holder
    <context:component-scan
    ..
    <tx:annotation-driven
    <mvc:annotation-drvent
    <aop:config

我们目前看基本标签的解析;

  • import标签

    markdown 复制代码
    可以引入其他的配置文件
    <import resource="applicationContext1.xml"/>
    <import resource="applicationContext2.xml"/>
    <import resource="applicationContext3.xml"/>
  • alias标签

    markdown 复制代码
    别名
    <bean id="product" name= "p"  class="xxxx.xxx.Product"/>
    	name代表别名
    也可以这样写
    <bean id="user"  class="xxxx.xxx.User"/>
    <alias name="user" alias="u"/>
    alias标签也可以写别名
  • beans标签上文已经讲过

  • bean标签------用的最多,着重分析

    markdown 复制代码
    BeanDefinitionHolder bdHolder = delegate.parseBeanDefinitionElement(ele);
    	解析标签里的元素,封装成BeanDefinitionHolder
        (对BeanDefinition做了一层包装)
        其中有三个属性
        private final BeanDefinition beanDefinition;
        存储BeanDefinition
    
        private final String beanName;
        存储id值,如果没有id,就是name值
        如果也没有name,会自动生成一个
    
        private final String[] aliases;
        存储别名
    	
    bdHolder = delegate.decorateBeanDefinitionIfRequired(ele, bdHolder);
    	如果有自定义标签,再解析自定义标签
    
    BeanDefinitionReaderUtils.registerBeanDefinition(bdHolder, getReaderContext().getRegistry());
    	将BeanDefinition,以id为key存储到map中
    
    getReaderContext().fireComponentRegistered(new BeanComponentDefinition(bdHolder));
    	发送注册完成的事件;此方法是空实现,可以自己实现
    parseBeanDefinitionElement重载1
    markdown 复制代码
    此方法用来解析id、别名,判断id是否唯一、调用parseBeanDefinitionElement重载2;若id,name都没有,生成id等操作。
parseBeanDefinitionElement重载2
markdown 复制代码
此方法用来解析class标签,创建BeanDefinition、解析scope、abstract标签等。
相关推荐
FG.2 分钟前
Day22
java·面试
菜鸟的迷茫4 分钟前
Redis 缓存雪崩、穿透、击穿面试题深度解析与 Spring Boot 实战代码示例
java
珹洺15 分钟前
C++算法竞赛篇:DevC++ 如何进行debug调试
java·c++·算法
SHUIPING_YANG22 分钟前
根据用户id自动切换表查询
java·服务器·数据库
爱吃烤鸡翅的酸菜鱼35 分钟前
IDEA高效开发:Database Navigator插件安装与核心使用指南
java·开发语言·数据库·编辑器·intellij-idea·database
惊涛骇浪、41 分钟前
SpringMVC + Tomcat10
java·tomcat·springmvc
墨染点香1 小时前
LeetCode Hot100【6. Z 字形变换】
java·算法·leetcode
ldj20201 小时前
SpringBoot为什么使用new RuntimeException() 来获取调用栈?
java·spring boot·后端
超龄超能程序猿1 小时前
Spring 应用中 Swagger 2.0 迁移 OpenAPI 3.0 详解:配置、注解与实践
java·spring boot·后端·spring·spring cloud
风象南1 小时前
SpringBoot配置属性热更新的轻量级实现
java·spring boot·后端