|
对于最新的稳定版本,请使用 Spring Framework 7.0.6! |
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 兼容(即,它必须接受父 Bean 的属性值)。
子Bean定义从父Bean继承作用域、构造函数参数值、属性值和方法覆盖,同时可以选择添加新值。您指定的任何作用域、初始化方法、销毁方法或<code>0</code>工厂方法设置都会覆盖相应的父设置。
其余的设置始终来自子定义:depends on、自动装配模式、依赖检查、单例和延迟初始化。
前面的示例通过使用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的ref属性引用,或使用父级Bean ID进行显式的getBean()调用,来单独使用这样的abstract父级Bean会返回错误。同样,容器的内部preInstantiateSingletons()方法会忽略被定义为抽象的Bean。
ApplicationContext 默认预先实例化所有单例。因此,对于单例 bean 来说,这一点很重要。如果你有一个(父)bean 定义,你只打算作为模板使用,并且此定义指定了一个类,那么你必须确保将 abstract 属性设置为 true,否则应用程序上下文实际上(会尝试)预先实例化 abstract 个 bean。 |