Spring的IoC容器之BeanFactory下

容器背后的秘密

除了了解Spring的IoC容器如何使用,了解Spring的IoC容器都提供了哪些功 能,我们也应该想一下,Spring的IoC容器内部到底是如何来实现这些的呢?虽然我们不太可能“重新 发明轮子”,但是,如图4-7(该图摘自Spring官方参考文档)所示的那样,只告诉你“Magic Happens Here”,你是否就能心满意足呢?


image.png

Spring的IoC容器所起的作用,就像图4-7所展示的那样,它会以某种方式加载Configuration Metadata(通常也就是XML格式的配置信息),然后根据这些信息绑定整个系统的对象,最终组装成 一个可用的基于轻量级容器的应用系统。

Spring的IoC容器实现以上功能的过程,基本上可以按照类似的流程划分为两个阶段,即容器启动 8 阶段和Bean实例化阶段,如图4-8所示。

Spring的IoC容器在实现的时候,充分运用了这两个实现阶段的不同特点,在每个阶段都加入了相 应的容器扩展点,以便我们可以根据具体场景的需要加入自定义的扩展逻辑。


image.png
  1. 容器启动阶段
    容器启动伊始,首先会通过某种途径加载Configuration MetaData。除了代码方式比较直接,在大 部分情况下,容器需要依赖某些工具类(BeanDefinitionReader)对加载的Configuration MetaData进行解析和分析,并将分析后的信息编组为相应的BeanDefinition,最后把这些保存了bean定义必 要信息的BeanDefinition,注册到相应的BeanDefinitionRegistry,这样容器启动工作就完成了。 图4-9演示了这个阶段的主要工作。
image.png

总地来说,该阶段所做的工作可以认为是准备性的,重点更加侧重于对象管理信息的收集。当然, 一些验证性或者辅助性的工作也可以在这个阶段完成。

  1. Bean实例化阶段

经过第一阶段,现在所有的bean定义信息都通过BeanDefinition的方式注册到了BeanDefini- tionRegistry中。当某个请求方通过容器的getBean方法明确地请求某个对象,或者因依赖关系容器 需要隐式地调用getBean方法时,就会触发第二阶段的活动。

该阶段,容器会首先检查所请求的对象之前是否已经初始化。如果没有,则会根据注册的 BeanDefinition所提供的信息实例化被请求对象,并为其注入依赖。如果该对象实现了某些回调接 口,也会根据回调接口的要求来装配它。当该对象装配完毕之后,容器会立即将其返回请求方使用。 如果说第一阶段只是根据图纸装配生产线的话,那么第二阶段就是使用装配好的生产线来生产具体的 产品了

插手“容器的启动”

Spring提供了一种叫做BeanFactoryPostProcessor的容器扩展机制。该机制允许我们在容器实 例化相应对象之前,对注册到容器的BeanDefinition所保存的信息做相应的修改。这就相当于在容 器实现的第一阶段最后加入一道工序,让我们对最终的BeanDefinition做一些额外的操作,比如修 改其中bean定义的某些属性,为bean定义增加其他信息等。

如果要自定义实现BeanFactoryPostProcessor,通常我们需要实现org.springframework. beans.factory.config.BeanFactoryPostProcessor接口。同时,因为一个容器可能拥有多个Bean- FactoryPostProcessor,这个时候可能需要实现类同时实现Spring的org.springframework.core. Ordered接口,以保证各个BeanFactoryPostProcessor可以按照预先设定的顺序执行(如果顺序紧 要的话)。但是,因为Spring已经提供了几个现成的BeanFactoryPostProcessor实现类,所以,大 多时候,我们很少自己去实现某个BeanFactoryPostProcessor。其中,org.springframework.beans. factory.config.PropertyPlaceholderConfigurer和org.springframework.beans.factory. config.Property OverrideConfigurer是两个比较常用的BeanFactoryPostProcessor。另外,为 了处理配置文件中的数据类型与真正的业务对象所定义的数据类型转换,Spring还允许我们通过 org.springframework.beans.factory.config.CustomEditorConfigurer 来 注 册 自 定 义 的 Pro- pertyEditor以补助容器中默认的PropertyEditor。

我们可以通过两种方式来应用BeanFactoryPostProcessor,分别针对基本的IoC容器 BeanFactory和较为先进的容器ApplicationContext。

对于BeanFactory来说,我们需要用手动方式应用所有的BeanFactoryPostProcessor,代码清单4-41演示了具体的做法。


image.png

如果拥有多个BeanFactoryPostProcessor,我们可以添加更多类似的代码来应用所有的这些 BeanFactoryPostProcessor。

对于ApplicationContext来说,情况看起来要好得多。因为ApplicationContext会自动识别配 置文件中的BeanFactoryPostProcessor并应用它,所以,相对于BeanFactory,在ApplicationContext 中加载并应用BeanFactoryPostProcessor,仅需要在XML配置文件中将这些BeanFactoryPost- Processor简单配置一下即可。只要如代码清单4-42所示,将相应BeanFactoryPostProcessor实现类 添加到配置文件,ApplicationContext将自动识别并应用它。

[图片上传中...(image.png-9a9729-1595433109837-0)]

下面让我们看一下Spring提供的这几个BeanFactoryPostProcessor实现都可以完成什么功能。

  1. PropertyPlaceholderConfigurer

通常情况下,我们不想将类似于系统管理相关的信息同业务对象相关的配置信息混杂到XML配置
文件中,以免部署或者维护期间因为改动繁杂的XML配置文件而出现问题。我们会将一些数据库连接 信息、邮件服务器等相关信息单独配置到一个properties文件中,这样,如果因系统资源变动的话,只 需要关注这些简单properties配置文件即可。

PropertyPlaceholderConfigurer允许我们在XML配置文件中使用占位符(PlaceHolder), 并将这些占位符所代表的资源单独配置到简单的properties文件中来加载。以数据源的配置为例,使用 了PropertyPlaceholderConfigurer之后(这里沿用代码清单4-42的配置内容),可以在XML配 置文件中按照代码清单4-43所示的方式配置数据源,而不用将连接地址、用户名和密码等都配置到 XML中。

image.png

如果你使用过Ant或者V elocity等工具,就会发现${property}之类的表达很熟悉。现在,所有这
些占位符所代表的资源,都放到了jdbc.properties文件中,如下所示:

image.png

基本机制就是之前所说的那样。当BeanFactory在第一阶段加载完成所有配置信息时,BeanFac- tory中保存的对象的属性信息还只是以占位符的形式存在,如{jdbc.url}、{jdbc.driver}。当 PropertyPlaceholderConfigurer作为BeanFactoryPostProcessor被应用时,它会使用properties 配置文件中的配置信息来替换相应BeanDefinition中占位符所表示的属性值。这样,当进入容器实 现的第二阶段实例化bean时,bean定义中的属性值就是最终替换完成的了。

PropertyPlaceholderConfigurer不单会从其配置的properties文件中加载配置项,同时还会检 查Java的System类中的Properties,可以通过setSystemPropertiesMode()或者setSystemProper- tiesModeName()来控制是否加载或者覆盖System相应Properties的行为。PropertyPlaceholder- Configurer提供了SYSTEM_PROPERTIES_MODE_FALLBACK、SYSTEM_PROPERTIES_MODE_NEVER和SYSTEM_ PROPERTIES_MODE_OVERRIDE三种模式。默认采用的是SYSTEM_PROPERTIES_ MODE_FALLBACK,即如果properties文件中找不到相应配置项,则到System的Properties中查找,我们还可以选择不检查System 的Properties或者覆盖它。

  1. PropertyOverrideConfigurer 。。。
  2. CustomEditorConfigurer
    其他两个BeanFactoryPostProcessor都是通过对BeanDefinition中的数据进行变更以达到某 种目的。与它们有所不同,CustomEditorConfigurer是另一种类型的BeanFactoryPostProcessor实 现,它只是辅助性地将后期会用到的信息注册到容器,对BeanDefinition没有做任何变动。

我们知道,不管对象是什么类型,也不管这些对象所声明的依赖对象是什么类型,通常都是通过 XML(或者properties甚至其他媒介)文件格式来配置这些对象类型。但XML所记载的,都是String类 型,即容器从XML格式的文件中读取的都是字符串形式,最终应用程序却是由各种类型的对象所构成。 要想完成这种由字符串到具体对象的转换(不管这个转换工作最终由谁来做),都需要这种转换规则 相关的信息,而CustomEditorConfigurer就是帮助我们传达类似信息的。

Spring内部通过JavaBean的PropertyEditor来帮助进行String类型到其他类型的转换工作。只要 为每种对象类型提供一个PropertyEditor,就可以根据该对象类型取得与其相对应的 PropertyEditor来做具体的类型转换。Spring容器内部在做具体的类型转换的时候,会采用JavaBean 框架内默认的PropertyEditor搜寻逻辑,从而继承了对原生类型以及java.lang.String.java.awt. Color和java.awt.Font等类型的转换支持。同时,Spring框架还提供了自身实现的一些Property- Editor,这些PropertyEditor大部分都位于org.springframework. beans.propertyeditors包 下。以下是这些Spring提供的部分PropertyEditor的简要说明。

image.png

以上这些PropertyEditor,容器通常会默认加载使用,所以,即使我们不告诉容器应该如何对 这些类型进行转换,容器同样可以正确地完成工作。但当我们需要指定的类型没有包含在以上所提到 的PropertyEditor之列的时候,就需要给出针对这种类型的PropertyEditor实现,并通过 CustomEditorConfigurer告知容器,以便容器在适当的时机使用到适当的PropertyEditor。

  • 自定义PropertyEditor
    通常情况下,对于Date类型,不同的Locale、不同的系统在表现形式上存在不同的需求。如系统这个部分需要以yyyy-MM-dd的形式表现日期,系统那个部分可能又需要以yyyyMMdd的形式对日期进 行转换。虽然可以使用Spring提供的CustomDateEditor,不过为了能够演示自定义PropertyEditor的 详细流程,在此我们有必要“重新发明轮子”!

下面是对自定义PropertyEditor实现的简单介绍。

假设需要对yyyy/MM/dd形式的日期格式转换提供支持。虽然可以直接让PropertyEditor实现类 去实现java.beans.PropertyEditor接口,不过,通常情况下,我们可以直接继承java.beans.Property- EditorSupport类以避免实现java.beans.PropertyEditor接口的所有方法。就好像这次,我们仅 仅 让 DatePropertyEditor 完 成 从 String 到 java.util.Date 的 转 换 , 只 需 要 实 现 setAsText(String)方法,而其他方法一概不管。该自定义PropertyEditor类定义如代码清单4-44 所示。

image.png

如果仅仅是支持单向的从String到相应对象类型的转换,只要覆写方法setAsText(String)即 9 可。如果想支持双向转换,需要同时考虑getAsText()方法的覆写。

通过CustomEditorConfigurer注册自定义的PropertyEditor

如果有类似于DateFoo这样的类对java.util.Date类型的依赖声明,通常情况下,会以代码清单 4-45所示的形式声明并将该类配置到容器中。

image.png

但是,默认情况下,Spring容器找不到合适的PropertyEditor将字符串“2007/10/16”转换成对象所声明的java.util.Date类型。所以,我们通过CustomEditorConfigurer将刚实现的DatePro- pertyEditor注册到容器,以告知容器按照DatePropertyEditor的形式进行String到java.util. Date类型的转换工作。

如果使用的容器是BeanFactory的实现,比如XmlBeanFactory,就需要通过编码手动应用 CustomEditorConfigurer到容器,类似如下形式:


image.png

但如果使用的是ApplicationContext相应实现,因为ApplicationContext会自动识别Bean- FactoryPostProcessor并应用,所以只需要在相应配置文件中配置一下,如代码清单4-46所示。

image.png
了解 bean 的一生

在已经可以借助于BeanFactoryPostProcessor来干预Magic实现的第一个阶段(容器启动阶段) 的活动之后,我们就可以开始探索下一个阶段,即bean实例化阶段的实现逻辑了。

容器启动之后,并不会马上就实例化相应的bean定义。我们知道,容器现在仅仅拥有所有对象的 BeanDefinition来保存实例化阶段将要用的必要信息。只有当请求方通过BeanFactory的getBean() 方法来请求某个对象实例的时候,才有可能触发Bean实例化阶段的活动。BeanFactory的getBean方 法可以被客户端对象显式调用,也可以在容器内部隐式地被调用。隐式调用有如下两种情况。

  • 对于BeanFactory来说,对象实例化默认采用延迟初始化。通常情况下,当对象A被请求而需 要第一次实例化的时候,如果它所依赖的对象B之前同样没有被实例化,那么容器会先实例化 对象A所依赖的对象。这时容器内部就会首先实例化对象B,以及对象 A依赖的其他还没有被 实例化的对象。这种情况是容器内部调用getBean(),对于本次请求的请求方是隐式的。

  • ApplicationContext启动之后会实例化所有的bean定义,这个特性已经多次提到。 但ApplicationContext在实现的过程中依然遵循Spring容器实现流程的两个阶段,只不过它 会在启动阶段的活动完成之后,紧接着调用注册到该容器的所有bean定义的实例化方法 getBean()。这就是为什么当你得到ApplicationContext类型的容器引用时,容器内所有对 象已经被全部实例化完成。不信你查一下类org.springframework.context.support. AbstractApplicationContext的refresh()方法。

之所以说getBean()方法是有可能触发Bean实例化阶段的活动,是因为只有当对应某个bean定义的getBean()方法第一次被调用时,不管是显式的还是隐式的,Bean实例化阶段的活动才会被触发,第二次被调用则会直接返回容器缓存的第一次实例化完的对象实例(prototype类型bean除外)。当getBean()方法内部发现该bean定义之前还没有被实例化之后,会通过createBean()方法来进行具体 的对象实例化,实例化过程如图4-10所示。


image.png

Spring容器将对其所管理的对象全部给予统一的生命周期管理,这些被管理的对象完全摆脱了原 来那种“new完后被使用,脱离作用域后即被回收”的命运。下面我们将详细看一看现在的每个bean 在容器中是如何走过其一生的。

  1. Bean的实例化与BeanWrapper
    容器在内部实现的时候,采用“策略模式(Strategy Pattern)”来决定采用何种方式初始化bean实例。 通常,可以通过反射或者CGLIB动态字节码生成来初始化相应的bean实例或者动态生成其子类。

org.springframework.beans.factory.support.InstantiationStrategy定义是实例化策略 的抽象接口,其直接子类SimpleInstantiationStrategy实现了简单的对象实例化功能,可以通过 反射来实例化对象实例,但不支持方法注入方式的对象实例化。CglibSubclassingInstantiation- Strategy继承了SimpleInstantiationStrategy的以反射方式实例化对象的功能,并且通过CGLIB 的动态字节码生成功能,该策略实现类可以动态生成某个类的子类,进而满足了方法注入所需的对象 实例化需求。默认情况下,容器内部采用的是CglibSubclassingInstantiationStrategy。

容器只要根据相应bean定义的BeanDefintion取得实例化信息,结合CglibSubclassingIns- tantiationStrategy以及不同的bean定义类型,就可以返回实例化完成的对象实例。但是,返回方 式上有些“点缀”。不是直接返回构造完成的对象实例,而是以BeanWrapper对构造完成的对象实例 进行包裹,返回相应的BeanWrapper实例。

至此,第一步结束。

BeanWrapper接口通常在Spring框架内部使用,它有一个实现类org.springframework.beans.BeanWrapperImpl。其作用是对某个bean进行“包裹”,然后对这个“包裹”的bean进行操作,比如设置或者获取bean的相应属性值。而在第一步结束后返回BeanWrapper实例而不是原先的对象实例, 就是为了第二步“设置对象属性”。

BeanWrapper定义继承了org.springframework.beans.PropertyAccessor接口,可以以统一的 方式对对象属性进行访问;BeanWrapper定义同时又直接或者间接继承了PropertyEditorRegistry 和TypeConverter接口。不知你是否还记得CustomEditorConfigurer?当把各种PropertyEditor注 册给容器时,知道后面谁用到这些PropertyEditor吗?对,就是BeanWrapper!在第一步构造完成 对象之后,Spring会根据对象实例构造一个BeanWrapperImpl实例,然后将之前CustomEditor- Configurer注册的PropertyEditor复制一份给BeanWrapperImpl实例(这就是BeanWrapper同时又 是PropertyEditorRegistry的原因)。这样,当BeanWrapper转换类型、设置对象属性值时,就不 会无从下手了。

使用BeanWrapper对bean实例操作很方便,可以免去直接使用Java反射API(Java Reflection API) 操作对象实例的烦琐。来看一段代码(见代码清单4-49),之后我们就会更加清楚Spring容器内部是如何设置对象属性的了!

image.png

我想有了BeanWrapper的帮助,你不会想直接使用Java反射API来做同样事情的。代码清单4-50 演示了同样的功能,即直接使用Java反射API是如何实现的(忽略了异常处理相关代码)。

image.png
  1. 各色的Aware接口
    当对象实例化完成并且相关属性以及依赖设置完成之后,Spring容器会检查当前对象实例是否实现了一系列的以Aware命名结尾的接口定义。如果是,则将这些Aware接口定义中规定的依赖注入给当前对象实例。

这些Aware接口为如下几个。

  • org.springframework.beans.factory.BeanNameAware。如果Spring容器检测到当前对象实 例实现了该接口,会将该对象实例的bean定义对应的beanName设置到当前对象实例。

  • org.springframework.beans.factory.BeanClassLoaderAware。如果容器检测到当前对 象实例实现了该接口,会将对应加载当前bean的Classloader注入当前对象实例。默认会使用 加载org.springframework.util.ClassUtils类的Classloader。

  • org.springframework.beans.factory.BeanFactoryAware。在介绍方法注入的时候,我们 提到过使用该接口以便每次获取prototype类型bean的不同实例。如果对象声明实现了 BeanFactoryAware接口,BeanFactory容器会将自身设置到当前对象实例。这样,当前对象 实例就拥有了一个BeanFactory容器的引用,并且可以对这个容器内允许访问的对象按照需要 进行访问。

以上几个Aware接口只是针对BeanFactory类型的容器而言,对于ApplicationContext类型的容 器,也存在几个Aware相关接口。不过在检测这些接口并设置相关依赖的实现机理上,与以上几个接 口处理方式有所不同,使用的是下面将要说到的BeanPostProcessor方式。不过,设置Aware接口这 一步与BeanPostProcessor是相邻的,把这几个接口放到这里一起提及,也没什么不可以的。

对于ApplicationContext类型容器,容器在这一步还会检查以下几个Aware接口并根据接口定义 设置相关依赖。

  • org.springframework.context.ResourceLoaderAware 。 ApplicationContext 实 现 了 Spring的ResourceLoader接口(后面会提及详细信息)。当容器检测到当前对象实例实现了 ResourceLoaderAware接口之后,会将当前ApplicationContext自身设置到对象实例,这样 当前对象实例就拥有了其所在ApplicationContext容器的一个引用。

  • org.springframework.context.ApplicationContextAware。如果ApplicationContext 容器检测到当前对象实现了ApplicationContextAware接口,则会将自身注入当前对象实例。

3. BeanPostProcessor

BeanPostProcessor的概念容易与BeanFactoryPostProcessor的概念混淆。但只要记住Bean-PostProcessor是存在于对象实例化阶段,而BeanFactoryPostProcessor则是存在于容器启动阶段, 这两个概念就比较容易区分了。

与BeanFactoryPostProcessor通常会处理容器内所有符合条件的BeanDefinition类似,Bean- PostProcessor会处理容器内所有符合条件的实例化后的对象实例。该接口声明了两个方法,分别在 两个不同的时机执行,见如下代码定义:

public interface BeanPostProcessor
{
Object postProcessBeforeInitialization(Object bean, String beanName) throws ➥ BeansException;
Object postProcessAfterInitialization(Object bean, String beanName) throws ➥ BeansException;
}

postProcessBeforeInitialization()方法是图4-10中BeanPostProcessor前置处理这一步将 会执行的方法,postProcessAfterInitialization()则是对应图4-10中BeanPostProcessor后置处 理那一步将会执行的方法。BeanPostProcessor的两个方法中都传入了原来的对象实例的引用,这为 我们扩展容器的对象实例化过程中的行为提供了极大的便利,我们几乎可以对传入的对象实例执行任 何的操作。

通常比较常见的使用BeanPostProcessor的场景,是处理标记接口实现类,或者为当前对象提供 代理实现。在图4-10的第三步中,ApplicationContext对应的那些Aware接口实际上就是通过Bean- PostProcessor的方式进行处理的。当ApplicationContext中每个对象的实例化过程走到BeanPost- Processor前置处理这一步时,ApplicationContext容器会检测到之前注册到容器的Application- ContextAwareProcessor这个BeanPostProcessor的实现类,然后就会调用其postProcessBefore- Initialization()方法,检查并设置Aware相关依赖。ApplicationContextAwareProcessor的post- ProcessBeforeInitialization()代码很简单明了,见代码清单4-51。

image.png

除了检查标记接口以便应用自定义逻辑,还可以通过BeanPostProcessor对当前对象实例做更多 的处理。比如替换当前对象实例或者字节码增强当前对象实例等。Spring的AOP则更多地使用 BeanPostProcessor来为对象生成相应的代理对象,如org.springframework.aop.framework. autoproxy.BeanNameAutoProxyCreator。我们将在Spring AOP部分详细介绍该类和AOP相关概念。

BeanPostProcessor是容器提供的对象实例化阶段的强有力的扩展点。为了进一步演示它的强大 威力,我们有必要实现一个自定义的BeanPostProcessor。

  • 自定义BeanPostProcessor
    假设系统中所有的IFXNewsListener实现类需要从某个位置取得相应的服务器连接密码,而且系统中保存的密码是加密的,那么在IFXNewsListener发送这个密码给新闻服务器进行连接验证的时候,首先需要对系统中取得的密码进行解密,然后才能发送。我们将采用BeanPostProcessor技术, 对所有的IFXNewsListener的实现类进行统一的解密操作。

(1) 标注需要进行解密的实现类
为了能够识别那些需要对服务器连接密码进行解密的IFXNewsListener实现,我们声明了接口 PasswordDecodable,并要求相关IFXNewsListener实现类实现该接口。PasswordDecodable接口声 明以及相关的IFXNewsListener实现类定义见代码清单4-52。

image.png

(2) 实现相应的BeanPostProcessor对符合条件的Bean实例进行处理

我们通过PasswordDecodable接口声明来区分将要处理的对象实例 1 ,当检查到当前对象实例实现了该接口之后,就会从当前对象实例取得加密后的密码,并对其解密。然后将解密后的密码设置回 当前对象实例。之后,返回的对象实例所持有的就是解密后的密码,逻辑如代码清单4-53所示。

image.png

(3) 将自定义的BeanPostProcessor注册到容器
只有将自定义的BeanPostProcessor实现类告知容器,容器才会在合适的时机应用它。所以,我 们需要将PasswordDecodePostProcessor注册到容器。

对于BeanFactory类型的容器来说,我们需要通过手工编码的方式将相应的BeanPostProcessor 注册到容器,也就是调用ConfigurableBeanFactory的addBeanPostProcessor()方法,见如下代码

image.png

对于ApplicationContext容器来说,事情则方便得多,直接将相应的BeanPostProcessor实现 类通过通常的XML配置文件配置一下即可。ApplicationContext容器会自动识别并加载注册到容器 的BeanPostProcessor,如下配置内容将我们的PasswordDecodePostProcessor注册到容器:

image.png

合理利用BeanPostProcessor这种Spring的容器扩展机制,将可以构造强大而灵活的应用系统。

4. InitializingBean和init-method

org.springframework.beans.factory.InitializingBean是容器内部广泛使用的一个对象生命周期标识接口,其定义如下:

public interface InitializingBean {
void afterPropertiesSet() throws Exception;
}

该接口定义很简单,其作用在于,在对象实例化过程调用过“BeanPostProcessor的前置处理” 之后,会接着检测当前对象是否实现了InitializingBean接口,如果是,则会调用其afterProper- tiesSet()方法进一步调整对象实例的状态。比如,在有些情况下,某个业务对象实例化完成后,还 不能处于可以使用状态。这个时候就可以让该业务对象实现该接口,并在方法afterPropertiesSet() 中完成对该业务对象的后续处理。

虽然该接口在Spring容器内部广泛使用,但如果真的让我们的业务对象实现这个接口,则显得 Spring容器比较具有侵入性。所以,Spring还提供了另一种方式来指定自定义的对象初始化操作,那就 是在XML配置的时候,使用<bean>的init-method属性。

通过init-method,系统中业务对象的自定义初始化操作可以以任何方式命名,而不再受制于 InitializingBean的afterPropertiesSet()。如果系统开发过程中规定:所有业务对象的自定义初 始化操作都必须以init()命名,为了省去挨个<bean>的设置init-method这样的烦琐,我们还可以通 过最顶层的<beans>的default-init-method统一指定这一init()方法名。

5. DisposableBean与destroy-method

当所有的一切,该设置的设置,该注入的注入,该调用的调用完成之后,容器将检查singleton类 型的bean实例,看其是否实现了org.springframework.beans.factory.DisposableBean接口。或 者其对应的bean定义是否通过<bean>的destroy-method属性指定了自定义的对象销毁方法。如果是, 就会为该实例注册一个用于对象销毁的回调(Callback),以便在这些singleton类型的对象实例销毁之前执行销毁逻辑。

与InitializingBean和init-method用于对象的自定义初始化相对应,DisposableBean和 destroy-method为对象提供了执行自定义销毁逻辑的机会。

最常见到的该功能的使用场景就是在Spring容器中注册数据库连接池,在系统退出后,连接池应该关闭,以释放相应资源。代码清单4-56演示了通常情况下使用destroy-method处理资源释放的数据 源注册配置。

image.png

不过,这些自定义的对象销毁逻辑,在对象实例初始化完成并注册了相关的回调方法之后,并不 会马上执行。回调方法注册后,返回的对象实例即处于使用状态,只有该对象实例不再被使用的时候, 才会执行相关的自定义销毁逻辑,此时通常也就是Spring容器关闭的时候。但Spring容器在关闭之前, 不会聪明到自动调用这些回调方法。所以,需要我们告知容器,在哪个时间点来执行对象的自定义销 毁方法。

对于BeanFactory容器来说。我们需要在独立应用程序的主程序退出之前,或者其他被认为是合 适的情况下(依照应用场景而定),如代码清单4-57所示,调用ConfigurableBeanFactory提供的 destroySingletons()方法销毁容器中管理的所有singleton类型的对象实例。

image.png

如果不能在合适的时机调用destroySingletons(),那么所有实现了DisposableBean接口的对 象实例或者声明了destroy-method的bean定义对应的对象实例,它们的自定义对象销毁逻辑就形同 虚设,因为根本就不会被执行!

对于ApplicationContext容器来说。道理是一样的。但AbstractApplicationContext为我们 提供了registerShutdownHook()方法,该方法底层使用标准的Runtime类的addShutdownHook()方 式来调用相应bean对象的销毁逻辑,从而保证在Java虚拟机退出之前,这些singtleton类型的bean对象 实例的自定义销毁逻辑会被执行。当然AbstractApplicationContext注册的shutdownHook不只是 调用对象实例的自定义销毁逻辑,也包括ApplicationContext相关的事件发布等,代码清单4-58演 示了该方法的使用。

image.png

同样的道理,在Spring 2.0引入了自定义scope之后,使用自定义scope的相关对象实例的销毁逻辑, 也应该在合适的时机被调用执行。不过,所有这些规则不包 prototype类型的bean实例,因为prototype 对象实例在容器实例化并返回给请求方之后,容器就不再管理这种类型对象实例的生命周期了。

至此,bean走完了它在容器中“光荣”的一生。

小结

Spring的IoC容器主要有两种,即BeanFactory和ApplicationContext。

我们从对比使用BeanFactory开发前后的差别开始,阐述了BeanFactory作为一个具体的IoC Service Provider,它是如何支持各种对象注册以及依赖关系绑定的。XML自始至终都是Spring的IoC容 14 器支持最完善的Configuration Metadata提供方式。所以,我们接着从XML入手,深入挖掘了 BeanFactory(以及ApplicationContext)的各种潜力。

对于充满好奇心的我们,不会只停留在会使用BeanFactory进行开发这一层面。所以,最后我们 又一起探索了BeanFactory(当然,也是ApplicationContext)实现背后的各种奥秘。BeanFactory 是Spring提供的基础IoC容器,但并不是Spring提供的唯一IoC容器。ApplicationContext构建于 BeanFactory之上,提供了许多BeanFactory之外的特性。后面,我们将一起走入Application- Context的世界。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,456评论 5 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,370评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,337评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,583评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,596评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,572评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,936评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,595评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,850评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,601评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,685评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,371评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,951评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,934评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,167评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,636评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,411评论 2 342