文章作者:Tyan
博客:noahsnail.com | CSDN | 简书
3.7 Bean定义继承
Bean定义中可以包含许多配置信息,包括构造函数参数,属性值和容器的特定信息例如初始化方法,静态工厂方法名等等。子定义继承父定义的配置信息。子定义可以覆盖一些值,或按需要添加一些其它值。使用父子bean定义可以保存许多类型。实际上,这是一种模板形式。
如果你以编程方式使用ApplicationContext
接口,那么子bean定义是通过ChildBeanDefinition
定义来表示的。大多数用户不在这个层级上使用它们,而是在一些像ClassPathXmlApplicationContext
中声明式的配置bean定义。当你使用基于XML的配置元数据时,你可以使用parent
特性来指明一个子bean定义。在特性值中指定父bean。
<bean id="inheritedTestBean" abstract="true"
class="org.springframework.beans.TestBean">
<property name="name" value="parent"/>
<property name="age" value="1"/>
</bean>
<bean id="inheritsWithDifferentClass"
class="org.springframework.beans.DerivedTestBean"
parent="inheritedTestBean" init-method="initialize">
<property name="name" value="override"/>
<!-- the age property value of 1 will be inherited from parent -->
</bean>
如果子bean定义中没有指定要使用的bean类,则使用父定义中的bean类,但也可以覆盖它。在后一种情况下(没有指定要用的bean类),子bean定义必须与父bean协作,也就是说,它必须接收父定义的属性值。
子bean定义可以继承作用域,构造函数参数值,属性值,可以重写父方法,可以选择添加新值。你指定的任何作用域,初始化方法,析构方法,和/或静态工厂方法设置将会覆盖对应的父设置。
其余的设置都是从子定义中获取:依赖关系,自动装配模式,依赖检查,单例,延迟初始化。
前面的例子使用抽象特性将父bean定义显式的标记为abstract
。如果父定义没有指定一个类,需要显式的将父bean定义为abstract
,形式如下:
<bean id="inheritedTestBeanWithoutClass" abstract="true">
<property name="name" value="parent"/>
<property name="age" value="1"/>
</bean>
<bean id="inheritsWithClass" class="org.springframework.beans.DerivedTestBean"
parent="inheritedTestBeanWithoutClass" init-method="initialize">
<property name="name" value="override"/>
<!-- age will inherit the value of 1 from the parent bean definition-->
</bean>
父bean不能实例化,因为它不完整,并且它被显式的标记为abstract
。当一个bean定义是abstract
时,它只能是一个纯粹的bean定义模板,作为一个为子定义服务的父定义。当试图使用一个abstract
父bean时,可以通过另一个bean的ref
属性来引用它或通过父bean的id为参数显式的调用getBean()
方法,会返回一个错误。类似的,容器内部的preInstantiateSingletons()
方法会忽略抽象bean定义。
默认情况下
ApplicationContext
会预实例化所有的单例。因此,如果你想有一个(父)bean定义只作为模板来使用,这个定义中指定了一个类,那你必须确保设置abstract
特性为true
,否则应用上下文会(试图)预实例化这个abstract
bean。
网友评论