跳到主要内容

Bean 定义继承

ChatGPT-4o-mini 中英对照 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"/>
<!-- the age property value of 1 will be inherited from parent -->
</bean>
xml
  • 注意 parent 属性。

子 bean 定义如果没有指定,则使用父定义中的 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>
xml

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

备注

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