分析Spring IoC源码(二)BeanFactory初始化

一、项目创建

  1. 目录结构:
    在这里插入图片描述
  2. pom.xml文件
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"  
  xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">  
  <modelVersion>4.0.0</modelVersion>  
  
  <groupId>org.study</groupId>  
  <artifactId>spring</artifactId>  
  <version>0.0.1-SNAPSHOT</version>  
  <packaging>jar</packaging>  
  
  <name>spring</name>  
  <url>http://maven.apache.org</url>  
  
  <properties>  
        <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>  
        <spring.version>3.2.5.RELEASE</spring.version>  
    </properties>  
  
  <dependencies>  
   <dependency>  
            <groupId>junit</groupId>  
            <artifactId>junit</artifactId>  
            <version>4.10</version>  
        </dependency>  
   <dependency>  
            <groupId>org.springframework</groupId>  
            <artifactId>spring-orm</artifactId>  
            <version>${spring.version}</version>  
        </dependency>  
        <dependency>  
            <groupId>org.springframework</groupId>  
            <artifactId>spring-expression</artifactId>  
            <version>${spring.version}</version>  
        </dependency>  
        <dependency>  
            <groupId>org.springframework</groupId>  
            <artifactId>spring-aop</artifactId>  
            <version>${spring.version}</version>  
        </dependency>  
        <dependency>  
            <groupId>org.springframework</groupId>  
            <artifactId>spring-beans</artifactId>  
            <version>${spring.version}</version>  
        </dependency>  
        <dependency>  
            <groupId>org.springframework</groupId>  
            <artifactId>spring-context</artifactId>  
            <version>${spring.version}</version>  
        </dependency>  
        <dependency>  
            <groupId>org.springframework</groupId>  
            <artifactId>spring-context-support</artifactId>  
            <version>${spring.version}</version>  
        </dependency>  
        <dependency>  
            <groupId>org.springframework</groupId>  
            <artifactId>spring-test</artifactId>  
            <version>${spring.version}</version>  
        </dependency>  
        <dependency>  
            <groupId>org.springframework</groupId>  
            <artifactId>spring-tx</artifactId>  
            <version>${spring.version}</version>  
        </dependency>  
  </dependencies>  
</project>  
  1. SayService.java
package org.study.spring.ioc;  
  
public class SayService {  
      
    public void say(){  
        System.out.println("I am Spring");  
    }  
  
}  
  1. SpringIoCTest.java
package org.study.spring.ioc;  
  
import org.junit.Test;  
import org.springframework.context.ApplicationContext;  
import org.springframework.context.support.ClassPathXmlApplicationContext;  
  
public class SpringIoCTest {  
      
    @Test  
    public void test1(){  
          
         ApplicationContext applicationContext = new ClassPathXmlApplicationContext("bean.xml");  
         SayService sayService = (SayService)applicationContext.getBean("test");  
         sayService.say();  
    }  
  
}  
  1. bean.xml
<?xml version="1.0" encoding="UTF-8"?>  
<beans xmlns="http://www.springframework.org/schema/beans"  
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:context="http://www.springframework.org/schema/context"  
    xmlns:mybatis-spring="http://mybatis.org/schema/mybatis-spring"  
    xmlns:tx="http://www.springframework.org/schema/tx" xmlns:jee="http://www.springframework.org/schema/jee"  
    xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.2.xsd  
        http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-3.2.xsd  
        http://www.springframework.org/schema/jee http://www.springframework.org/schema/jee/spring-jee-3.2.xsd  
        http://mybatis.org/schema/mybatis-spring http://mybatis.org/schema/mybatis-spring-1.2.xsd  
        http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-3.2.xsd">  
   
      
    <bean id="test" class="org.study.spring.ioc.SayService"></bean>  
  
</beans>  

运行结果:
在这里插入图片描述
好了,我们以Debug模式在SpringIoCTest的第12行ClassPathXmlApplicationContext一行打上断点
在Debug之前,我们先把ClassPathXmlApplicationContext的继承图列一下
在这里插入图片描述
附一张BeanDefinition核心类图
在这里插入图片描述
好了,一切准备工作好了之后,开始以debug模式运行
在这里插入图片描述

打开“this”构造函数
在这里插入图片描述
进入137行,看setConfigLocations要对我们的配置文件“bean.xml“做什么,打开setConfigLocations
在这里插入图片描述
上图中this.configureLations指的就是图一中用红旗②标记的configLactions,这步就理解了,给AbstractRefreshableConfigApplicationContext.Java中的configLocations赋值
setConfigLocations完后,图三第139行需要进行refresh(),进入refresh(),发现refresh方法在AbstractApplicationContext.java中,也就是图一编辑的”①“标记的那个方法,打开
在这里插入图片描述
refresh,如图:
今天我们主要看的就是spring如何初始化beanfactory的,红色区域已经标记出,进入obtainFreshBeanFactory这个
方法
在这里插入图片描述

obtainFreshBeanFactory这个方法里面主要有2个步骤,第一个步骤537行的refreshBeanFactory(),
在这里插入图片描述
122-125行表示如果已经有了一个容器,先销毁里面的bean然后再关闭容器,spring保证只有一个容器,然后我们看131~133行,在线程安全的状态下,为this.beanfactory赋值,那么this.beanfactory是什么呢,就是图一中红旗①标记的DefaultListableBeanFactory beanFactory,现在调用方法的顺序也如图一中用①②③这种标记已经标出,refreshBeanFactory()这个方法的最终目的就是为AbstractRefreshableApplicationContext.java中DefaultListableBeanFactory beanFactory赋值,那么127-130行就是为beanFactory初始化这个容器里面的各个组件了,我们先暂时不分析127~130行代码,回到图四中538行代码,getBeanFactory(),打开getBeanFactory()

在这里插入图片描述

在线程安全的情况下,返回this.beanFactory,这个beanFactory是谁呢,原来就是刚刚refreshBeanFactory()方法赋值的那个AbstractRefreshableApplicationContext.java中的DefaultListableBeanFactory beanFactory,现在一切都顺理成章了,beanfactory就是这么初始化好的,然后返回的
现在我们回头分析一下图五中127~130行行代码,看看beanfactory返回之前,自己完成了哪些初始化动作
127行,先创建一个beanfactory,128行设置序列化Id,129行定制化beanfactory,130行看到了我们最想看到的词”beanDefinition‘,这边终于开始加载beandefinnition了

打开loadBeanDefinitions这个方法
在这里插入图片描述

第82行,创建一一个XmlBeanDefinitionReader,并将beanFactory传了进去,上一节我们讲过beanFactory实现了BeanDefinitionRegistry这个接口,这时传入beanFactory,就是依据这个属性传入的
在这里插入图片描述

传入后,等beanDefinition好了之后就可以BeanDefinitionRegistry(beanfactory).registerBeanDefinition()了,这些都是我们上个章节讲过的,所以这里beanfactory有2种身份,接下来我们看86~92行代码,这边是对beanDefinitionReader做的一些准备初始化工作,我们不做深究,进入93行代码loadBeanDefinitions(beanDefinitionReader)
在这里插入图片描述
getConfigLocations这个方法我们研究过,发现它返回的正好是我们图三中setConfigLocations赋值的configLocations,即是我们传入的“bean.xml”
spring开始用我们配置的bean,xml开始初始化BeanDefinition,这边调用关系可以查看图二中标记的①②③④
最终找到XmlBeanDefinitionReader.java 中的registerBeanDefinitions(Document doc, Resource resource)方法
在这里插入图片描述

这个方法是返回新加入的beanDefinition的个数getRegistry()就是获取beanfactory因为beanfactory实现了BeanDefinitionRegistry,所以就可以调用getBeanDefinitionCount这个方法,我们分析493行代码registerBeanDefinitions(Document doc, XmlReaderContext readerContext),debug到DefaultBeanDefinitionDocumentReader.java中的doRegisterBeanDefinitions(Element root)这个方法
在这里插入图片描述
接着查看
在这里插入图片描述

循环bean,xml中定义的每一个元素,bean.xml中有些是标签是等等,分别解析在这里插入图片描述
我们这边只定义了一个
在这里插入图片描述
这边已经看出了传入了bean.xml中的id叫做test的bean,它的class是,并且也传入了beanfactory(beanDefinition),感觉万事具备,大家看registerBeanDefinition这个方法
在这里插入图片描述

在这里插入图片描述
最后查看该bean是否有别名,如果有别名也一起注册一下,防止用户根据别名去获取bean
在这里插入图片描述

因为我们没有为test设置别名,所以该aliases为null,大家可以自己设置一下

好了,到此为止,我们应该知道beanFactory是怎么初始化的了,也知道beanFactory如何去加载bean.xml中的的bean了,分析的不是很好,我觉得最关键的就是图一图二显示的,理解好各个属性的关系和赋值,最后知道BeanFactory有2个角色,这样就方便大家理解Spring是如何初始化的了,大家还是自己动手理解一下吧,希望对大家有帮助~

接下来的几个章节中,我们一起理解一下依赖注入,和refresh()这个核心方法其他的方法吧
在这里插入图片描述

猜你喜欢

转载自blog.csdn.net/weixin_43397326/article/details/83108361