6.7 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">
<property name="name" value="override"/>
<!-- the age property value of 1 will be inherited from parent -->
</bean>
如果没有指定,则子bean的定义使用父定义的bean类,但也可以覆盖它。在后一种情况下,子bean类必须与父类兼容,也就是说,它必须接受父类的属性值。
子bean的定义会从父亲那里继承范围,构造函数参数值,属性值和覆写方法,并带有添加新值的选项。您指定的任何范围,初始化方法,销毁方法和/或静态工厂方法设置都将覆盖相应的父设置。
其余设置始终取自子定义:depends on,autowire模式,依赖性检查,singleton,lazy init。
前面的示例通过使用abstract属性将父bean定义显式标记为abstract。如果父定义未指定类,则需要将父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,通过将它作为另一个bean的ref属性引用或者使用父bean的id进行显式的getBean()调用,会返回一个错误。类似地,容器的内部preInstantiateSingletons()方法会忽略定义为abstract的bean定义。
ApplicationContext默认情况下预先实例化所有单例。因此,重要的是(至少对于单例bean),如果你有一个(父)bean定义,你只打算用作模板,并且这个定义指定了一个类,你必须确保将abstract属性设置为true 否则应用程序上下文将实际(尝试)预先实例化抽象bean。