容器背后的秘密
了解 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所示。
Spring容器将对其所管理的对象全部给予统一的生命周期管理,这些被管理的对象完全摆脱了原来那种“new完后被使用,脱离作用域后即被回收”的命运。
下面我们将详细看一看现在的每个bean在容器中是如何走过其一生的。
可以在org.springframework.beans.factory.support.AbstractBeanFactory类的代码中查看到getBean()方法的完整实现逻辑,可以在其子类org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory的代码中一窥createBean()方法的全貌。
1. Bean的实例化与BeanWrapper
容器在内部实现的时候,采用“策略模式(StrategyPattern
)”来决定采用何种方式初始化bean实例。通常,可以通过反射或者CGLIB动态字节码 生成来初始化相应的bean实例或者动态生成其子类。
org.springframework.beans.factory.support.InstantiationStrategy
定义是实例化策略的抽象接口,其直接子类SimpleInstantiationStrategy
实现了简单的对象实例化功能,可以通过反射来实例化对象实例,但不支持方法注入方式的对象实例化。
CglibSubclassingInstantiationStrategy
继承了SimpleInstantiationStrategy
的以反射方式实例化对象的功能,并且通过CGLIB的动态字节码生成功能,该策略实现类可以动态生成某个类的子类,进而满足了方法注入所需的对象实例化需求。默认情况下,容器内部采用的是CglibSubclassingInstantiationStrategy
。
容器只要根据相应bean定义的BeanDefintion取得实例化信息,结合CglibSubclassingInstantiationStrategy
以及不同的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
实例,然后将之前CustomEditorConfigurer
注册的PropertyEditor
复制一份给BeanWrapperImpl
实例(这就是BeanWrapper
同时又是PropertyEditorRegistry
的原因)。这样,当BeanWrapper
转换类型、设置对象属性值时,就不会无从下手了。
使用BeanWrapper
对bean实例操作很方便,可以免去直接使用Java反射API(Java Reflection
API)操作对象实例的烦琐。来看一段代码,之后我们就会更加清楚Spring容器内部是如何设置对象属性的了!
我想有了BeanWrapper的帮助,你不会想直接使用Java反射API来做同样事情的。下方代码演示了同样的功能,即直接使用Java反射API是如何实现的(忽略了异常处理相关代码)。
如果你觉得没有太大差别,那是因为没有看到紧随其后的那些异常(exception)还有待处理!
2. 各色的 Aware 接口
当对象实例化完成并且相关属性以及依赖设置完成之后,Spring容器会检查当前对象实例是否实现了一系列的以Aware命名结尾的接口定义。如果是,则将这些Aware接口定义中规定的依赖注入给当前对象实例。
这里不继续介绍了,更多看原文吧
3. BeanPostProcessor
BeanPostProcessor
的概念容易与BeanFactoryPostProcessor
的概念混淆。但只要记住BeanPostProcessor
是存在于对象实例化阶段,而BeanFactoryPostProcessor
则是存在于容器启动阶段,这两个概念就比较容易区分了。
与BeanFactoryPostProcessor
通常会处理容器内所有符合条件的BeanDefinition
类似,BeanPostProcessor
会处理容器内所有符合条件的实例化后的对象实例。该接口声明了两个方法,分别在两个不同的时机执行,见如下代码定义:
postProcessBeforeInitialization()
方法是图4-10中BeanPostProcessor
前置处理这一步将会执行的方法,postProcessAfterInitialization()
则是对应图4-10中BeanPostProcessor
后置处理那一步将会执行的方法。BeanPostProcessor
的两个方法中都传入了原来的对象实例的引用,这为我们扩展容器的对象实例化过程中的行为提供了极大的便利,我们几乎可以对传入的对象实例执行任何的操作。
通常比较常见的使用BeanPostProcessor
的场景,是 处理标记接口实现类,或者为当前对象提供代理实现。在图4-10的第三步中,ApplicationContext
对应的那些Aware接口实际上就是通过BeanPostProcessor
的方式进行处理的。
当ApplicationContext
中每个对象的实例化过程走到BeanPostProcessor
前置处理这一步时,ApplicationContext
容器会检测到之前注册到容器的ApplicationContextAwareProcessor
这个BeanPostProcessor
的实现类,然后就会调用其postProcessBeforeInitialization()
方法,检查并设置Aware相关依赖。
ApplicationContextAwareProcessor
的postProcessBeforeInitialization()
代码很简单明了,见下方代码。
除了检查标记接口以便应用自定义逻辑,还可以通过BeanPostProcessor
对当前对象实例做更多的处理。比如替换当前对象实例或者字节码增强当前对象实例等。Spring的AOP则更多地使用BeanPostProcessor
来为对象生成相应的代理对象,如org.springframework.aop.framework.autoproxy.BeanNameAutoProxyCreator
。我们将在SpringAOP部分详细介绍该类和AOP相关概念。
BeanPostProcessor
是容器提供的对象实例化阶段的强有力的扩展点。为了进一步演示它的强大威力,我们有必要实现一个自定义的BeanPostProcessor
。
自定义BeanPostProcessor
假设系统中所有的IFXNewsListener
实现类需要从某个位置取得相应的服务器连接密码,而且系统中保存的密码是加密的,那么在IFXNewsListener
发送这个密码给新闻服务器进行连接验证的时候,首先需要对系统中取得的密码进行解密,然后才能发送。我们将采用BeanPostProcessor
技术,对所有的IFXNewsListener
的实现类进行统一的解密操作。
(1) 标注需要进行解密的实现类
为了能够识别那些需要对服务器连接密码进行解密的IFXNewsListener
实现,我们声明了接口PasswordDecodable
,并要求相关IFXNewsListener
实现类实现该接口。PasswordDecodable
接口声明以及相关的IFXNewsListener
实现类定义见下方代码。
(2) 实现相应的BeanPostProcessor对符合条件的Bean实例进行处理
我们通过PasswordDecodable
接口声明来区分将要处理的对象实例,当检查到当前对象实例实现了该接口之后,就会从当前对象实例取得加密后的密码,并对其解密。然后将解密后的密码设置回当前对象实例。之后,返回的对象实例所持有的就是解密后的密码,逻辑如下方代码所示。
(3) 将自定义的BeanPostProcessor注册到容器
只有将自定义的BeanPostProcessor
实现类告知容器,容器才会在合适的时机应用它。所以,我们需要将PasswordDecodePostProcessor
注册到容器。
对于BeanFactory
类型的容器来说,我们需要通过手工编码的方式将相应的BeanPostProcessor
注册到容器,也就是调用ConfigurableBeanFactory
的addBeanPostProcessor()
方法,见如下代码
对于ApplicationContext
容器来说,事情则方便得多,直接将相应的BeanPostProcessor
实现类通过通常的XML配置文件配置一下即可。ApplicationContext
容器会自动识别并加载注册到容器的BeanPostProcessor
,如下配置内容将我们的PasswordDecodePostProcessor
注册到容器:
合理利用BeanPostProcessor
这种Spring的容器扩展机制,将可以构造强大而灵活的应用系统。