https://docs.spring.io/spring-framework/reference/core/beans/child-bean-definitions.html
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定义继承父定义中的scope、构造函数参数值、属性值和方法覆盖,可以选择添加新的值。你指定的任何scope、初始化方法、销毁方法或静态工厂方法设置都会覆盖相应的父设置。
其余设置总是取自子定义:depends on、autowire模式、依赖性检查、singleton和lazy init。
前面的示例通过使用abstract
属性明确地将父bean定义标记为抽象。如果父定义没有指定类,那么必须显式地将父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
。当一个定义是abstract
的时,它只能作为一个纯模板bean定义使用,作为子定义的父定义。尝试单独使用这样的抽象父bean,通过将另一个bean的ref
属性引用到它,或者使用父bean ID进行显式的getBean()调
用都会返回错误。同样,容器的内部preInstantiateSingletons()
方法会忽略被定义为abstract
的bean定义。
ApplicationContext
默认预实例化所有singletons。因此,如果你有一个(父)bean定义,你打算仅将其用作模板,而这个定义指定了一个类,那么你必须确保将abstract
属性设置为true
,否则应用程序上下文实际上会(尝试)预实例化abstract
bean。