Spring:容器基本用法

时间:2023-03-08 21:44:28

bean是Spring 最核心的东西,打个比方,假设Spring是一个水桶,那么bean就是水桶里的水,水桶离开水后,就没啥作用了。我们先来看一下bean的定义:

 public class Person {
 private String name;
 private int age;

 public String getName() {
 return name;
 }

 public void setName(String name) {
 this.name = name;
 }

 public int getAge() {
 return age;
 }

 public void setAge(int age) {
 this.age = age;
 }

 public void info() {
 System.out.println("name:" + getName() + " age:" + getAge());
 }
 }

bean

这个bean 很普通?没错,Spring追求的是使我们的bean变成一个纯粹的POJO。

接下来看一下配置文件:

 <?xml version="1.0" encoding="UTF-8"?>
 <beans xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
     xmlns="http://www.springframework.org/schema/beans"
     xsi:schemaLocation="http://www.springframework.org/schema/beans
     http://www.springframework.org/schema/beans/spring-beans-3.0.xsd">
     <bean id="person" class="com.juin.entity.Person">
         <property name="name" value="xingoo"/>
         <property name="age" value="12"/>
     </bean>
 </beans>

bean.xml

在上面的配置文件中,我们看到了bean的声明方式。尽管Spring中的bean 有N多属性来支撑我们业务中的各种应用,但是只要我们声明成这样,那么一般都可以满足。下面就是测试代码啦:

 public class Test {

     public static void main(String[] args) {
         // TODO Auto-generated method stub
         ApplicationContext ctx = new ClassPathXmlApplicationContext("bean.xml");// 读取bean.xml中的内容
         Person p = ctx.getBean("person", Person.class);// 创建bean的引用对象
         p.info();
     }

 }

Test

相信聪明的人,应该能猜到 控制台里将会打印  name:xingoo age:12  了。

好了,到这里,我们又复习了一遍Spring,我相信很多人对此都感到不屑吧!哈哈,那你们想知道这个过程具体是如何实现的吗?那就要好好理解Spring内部的原理了。由于Spring是一个开源的框架,所以我们研究起来很方便,只要上网拉份源码下来,就可以来啃啦!我也是第一次读源码吧,为了让自己坚持下去,就计划没读一点,就逼着自己写一篇博客,这样能让自己更加与动力一点吧!接下来我们开始吧!

我们先来分析一下上面的测试代码:ApplicationContext ctx = new ClassPathXmlApplicationContext("bean.xml");

这段测试代码无非就是分为三个步骤:

1.读取bean.xml文件;

2.根据bean.xml的配置信息找到对应的类的配置并实例化;

3.调用实例化的实例;

为了更清楚地描述,请看下图:

Spring:容器基本用法

ConfigReader:用于读取和验证配置文件。我们要用配置文件里面的东西,当然首先要做的是读取,然后放置在内存中。

ReflectionUtil:用于根据配置文件中的配置文件进行反射实例化。比如上面的bean.xml中的<bean id="person" class="com.juin.entity.Person"/>,我们就可以根据bean.Person进行实例化。

App:用于完成整个逻辑的串联。

按照原始的思维方式,整个过程无非如此,但是作为一个风靡世界的优秀源码真的这么简单吗?答案当然是否定的。不妨一起来看看Spring的源码吧,用于实现上面功能的是org.Springframework.beans.jar。

Spring:容器基本用法

这个是gradle管理的,如果想用eclipse打开,可以安装一个gradle进行转换。

在开始分析源码之前,我们必须先来熟悉下Spring两个最核心的类:DefaultListableBeanFactory和XmlBeanDefinitionReader。

1.DefaultListableBeanFactory

XmlBeanFactory继承自DefaultListableBeanFactory,而DefaultListableBeanFactory是整个bean加载的核心部分,是Spring注册和加载bean的默认实现,而对于XmlBeanFactory与DefaultListableBeanFactory的不同之处在于XmlBeanFactory中使用了自定义的XML读取器XmlBeanDefinitionReader,实现了个性化的BeanDefinitionReader读取,DefaultListableBeanFactory继承了AbstractAutowireCapableBeanFactory并实现了ConfigurableListableBeanFactory以及BeanDefinitionRegistry接口。下面是ConfigurableListableBeanFactory的层次结构图以及相关类图。

Spring:容器基本用法

我先简单说一下上图中各类的作用吧:

1.AliasRegistry:定义对alias的简单增删改等操作。

2.SimpleAliasRegistry:主要使用map作为alias的缓存,并对接口AliasRegistry进行实现。

3.SingletonBeanRegistry:定义对单例的注册和获取。

4.BeanFactory:定义获取bean以及bean的各种属性。

5.DefaultSingletonBeanRegistry:对接口SingletonBeanRegistry各函数的实现。

6.HierarchicalBeanFactory:继承BeanFactory,也就是在BeanFactory定义的功能的基础上增加了对parentFactory的支持。

7.BeanDefinitionRegistry:定义对BeanDefinition的各种增删改操作。

8.FactoryBeanRegistrySupport:在DefaultSingletonBeanRegistry基础上增加了对FactoryBean的特殊处理功能。

9.ConfigurableBeanFactory:根据配置Factory的各种方法。

10.ListableBeanFactory:根据各种条件获取bean的配置清单。

11.AbstractBeanFactory:综合FactoryBeanRegistrySupport和ConfigurableBeanFactory的功能。

12.AutowireCapableBeanFactory:提供创建bean,自动注入,初始化以及应用bean的后处理器。

13.AbstractAutowireCapableBeanFactory:综合AbstractBeanFactory并对接口AutowireCapableBeanFactory进行实现。

14.ConfigurableListableBeanFactory:BeanFactory配置清单,指定忽略类型以及接口等。

15.DefaultListableBeanFactory:综合上面所有功能,主要是对Bean注册后的处理。

XmlBeanFactory对DefaultListableBeanFactory类进行了扩展,主要用于从XML文档中读取BeanDefinition,对于注册以及获取Bean都是使用从父类DefaultListableBeanFactory继承的方法去实现,而纬度与父类不同的个性化实现就是增加了XmlBeanDefinitionReader类型的reader属性。在XmlBeanFactory中主要使用reader属性对资源文件进行读取和注册。

2.XmlBeanDefinitionReader

Spring:容器基本用法

XML配置文件的读取是Spring中重要的功能,因为Spring的大部分功能都是以配置作为切入点的,那么我们可以从XmlBeanDefinitionReader中梳理一下资源文件读取,解析以及注册的大致脉络,首先我们可以看看各个类的功能。

1.ResourceLoader:定义资源加载器,主要用于根据给定的资源文件地址返回对应的Resource。

2.BeanDefinitionReader:主要定义资源文件读取并转换为BeanDefinition的各个功能。

3.EnvironmentCapable:定义获取Environment方法。

4.DocumentLoader:定义从资源文件加载到转换为Document的功能。

5.AbstractBeanDefinitionReader:对EnvironmentCapable,BeanDefinitionReader类定义的功能的实现。

6.BeanDefinitionDocumentReader:定义读取Document并注册BeanDefinition功能。

7.BeanDefinitionParserDelegate:定义解析Element的各种方法。

通过上图我们可以知道XmlBeanDefinitionReader主要包括以下几步的处理:

(1)通过继承自AbstractBeanDefinitionReader中的方法,来使用ResourceLoader将资源文件路径转换为对应的Resource文件。

(2)通过DocumentLoader对Resource文件进行转换,将Resource文件转换为Document文件。

(3)通过实现接口BeanDefinitionDocumentReader的DefaultBeanDefinitionDocumentReader类对Document进行解析,并使用BeanDefinitionParserDelegate对Element进行解析。

3.容器的基础XmlBeanFactory

Spring:容器基本用法

到这里,我们已经对Spring的容器功能有了一个大致的了解。接下来我们会详细探索每个步骤的实现。接下来要深入分析以下功能的代码实现:

BeanFactory bf = new XmlBeanFactory(new ClassPathResource(bean.xml));

首先调用ClassPathResource的构造函数来构造Resource资源文件的实例对象,这样后续的资源处理就可以用Resource提供的各种服务来操作了,当我们有了Rssource后就可以进行XmlBeanFactory的初始化了。那么Resource资源是如何封装的呢?请看下一章节,