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
工厂方法设置都会覆盖相应的父设置。
其余设置始终取自子定义:depends-on、自动装配模式、依赖检查、singleton 和延迟初始化。
前面的示例通过使用 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 默认会预先实例化所有 singleton Bean。因此,重要的一点(至少对于 singleton Bean 来说)是,如果你有一个(父)Bean 定义仅打算用作模板,并且此定义指定了类,则必须确保将 abstract 属性设置为 true,否则应用上下文将实际(尝试)预先实例化该 abstract Bean。 |