Bean 定义继承

Bean 定义可以包含许多配置信息,包括构造函数参数、属性值和容器特定的信息,例如初始化方法、静态工厂方法名称等等。子 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"/>
	<!-- the age property value of 1 will be inherited from parent -->
</bean>
1 请注意 parent 属性。

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

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

其余设置始终从子定义中获取:依赖项、自动装配模式、依赖项检查、单例和延迟初始化。

前面的示例使用 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 定义,作为子定义的父定义。尝试单独使用此类 abstract 父 Bean,通过将其引用为另一个 Bean 的 ref 属性或对父 Bean ID 执行显式 getBean() 调用,会导致错误。类似地,容器的内部 preInstantiateSingletons() 方法会忽略定义为抽象的 Bean 定义。

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