Bean Definition Inheritance

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">  (1)
	<property name="name" value="override"/>
	<!-- 年龄属性值为1将从父bean继承 -->
</bean>
1 注意parent属性。

如果未指定子bean定义的bean类,则子bean定义将使用父定义的bean类,但也可以覆盖它。在后一种情况下,子bean类必须与父类兼容(即,它必须接受父类的属性值)。

子bean定义从父类继承作用域、构造函数参数值、属性值和方法覆盖,同时可以添加新值。您指定的任何作用域、初始化方法、销毁方法或static工厂方法设置都会覆盖相应的父设置。

其余设置始终从子定义中获取: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"/>
	<!-- 年龄将从父bean定义继承值为1 -->
</bean>

父bean不能单独实例化,因为它是不完整的,而且还明确标记为abstract。当一个定义是abstract时,它只能用作纯模板bean定义,作为子定义的父定义。尝试单独使用这样的abstract父bean,通过将其作为另一个bean的ref属性引用或使用父bean ID进行显式getBean()调用会返回错误。同样,容器的内部preInstantiateSingletons()方法会忽略被定义为抽象的bean定义。

ApplicationContext默认预先实例化所有单例。因此,对于单例bean(至少对于单例bean),如果您有一个(父)bean定义,您只打算将其用作模板,并且此定义指定了一个类,则必须确保将abstract属性设置为true,否则应用程序上下文实际上(尝试)预先实例化abstract bean。