例如,一个bean,它使用一个配置了值12的协作者bean
<bean name="beanService12" class="SomeSevice">
<constructor-arg index="0" ref="serviceCollaborator1"/>
</bean>
<bean name="serviceCollaborator1" class="SomeCollaborator">
<constructor-arg index="0" value="12"/>
<!-- more cargs,more beans,more flavor -->
</bean>
然后,我希望能够创建类似的bean,配置的协作者略有不同.我可以做些什么吗
<bean name="beanService13" parent="beanService12">
<constructor-arg index="0">
<bean>
<constructor-arg index="0" value="13"/>
</bean>
</constructor>
</bean>
我不知道这是可能的,如果是这样,它感觉有点笨拙.是否有更好的方法来覆盖大型嵌套bean定义的小部分?看起来孩子豆对于父母来说已经知道了很多,例如构造器索引.
这是一个玩具示例 – 在实践中,服务是一个大型bean定义,依赖于许多其他协作者bean,其中还有其他bean依赖关系.例如,创建一个处理程序链,每个bean引用链中的下一个,引用下一个.我想创建一个几乎相同的链条,中间的处理程序有一些小的变化,我该怎么办?
我不想更改结构 – 服务bean使用协作者来执行其功能,但如果有帮助,我可以添加属性并使用属性注入.
这是一个重复的模式,会创建一个自定义模式帮助?
感谢任何建议!
编辑:我的问题的结论是,如果我有一个非常大的bean定义,创建一个复杂的bean的创建(bean具有bean等等),我想创建一个几乎相同的bean几个变化,我该怎么办?请注意,如果您的解决方案必须使用属性,或者可以使用构造函数注入.
嵌套和顶级bean不是问题(实际上,我认为所有的bean都是实践中的顶级).
EDIT2:谢谢你的答复.一个factorybean可能是一个答案,因为这将减少弹簧上下文的复杂性,并允许我仅仅将差异指定为工厂的参数.但是,将一大堆上下文重新编入代码并不正确.我听说过春天可以用脚本,例如groovy – 是否提供了另一种选择?工厂可以在groovy中创建吗?
解决方法
首先,定义一个抽象Bean作为外部bean的模板(我的示例使用一个Car作为外部bean,一个Engine作为内部bean),给出所有其他bean可以继承的默认值:
<bean id="defaultCar" class="Car" abstract="true">
<property name="make" value="Honda"/>
<property name="model" value="Civic"/>
<property name="color" value="Green"/>
<property name="numberOfWheels" value="4"/>
<property name="engine" ref="defaultEngine"/>
</bean>
由于所有本田思域都具有相同的引擎(在我的世界,我对车无所知),我给它一个默认的嵌套引擎bean.不幸的是,一个bean不能引用抽象Bean,所以默认引擎不能是抽象的.我为引擎定义了一个具体的bean,但将其标记为lazy-init,因此除非另有一个bean使用它,否则实际上不会实例化它:
<bean id="defaultEngine" class="Engine" lazy-init="true">
<property name="numberOfCylinders" value="4"/>
<property name="volume" value="400"/>
<property name="weight" value="475"/>
</bean>
现在我可以定义我的特定车,通过引用通过父级定义的bean来获取所有默认值:
<bean id="myCar" parent="defaultCar"/>
我的妻子有一辆像我的车,除了它的一个不同的模型(再一次,我不知道汽车 – 让我们假设发动机是一样的,即使在现实生活中,他们可能不是).而不是重新定义一堆bean /属性,我只是再次扩展默认的汽车定义,但是覆盖其一个属性:
<bean id="myWifesCar" parent="defaultCar">
<property name="model" value="Odyssey"/>
</bean>
我妹妹和我妻子有同样的车(真的),但它有不同的颜色.我可以扩展一个具体的bean并覆盖其上的一个或多个属性:
<bean id="mySistersCar" parent="myWifesCar">
<property name="color" value="Silver"/>
</bean>
如果我喜欢赛车小型货车,我可能会考虑用一个更大的发动机.在这里我扩展一个小型货车,用一个新的引擎覆盖它的默认引擎.这个新引擎扩展了默认引擎,覆盖了一些属性:
<bean id="supedUpMiniVan" parent="myWifesCar">
<property name="engine">
<bean parent="defaultEngine">
<property name="volume" value="600"/>
<property name="weight" value="750"/>
</bean>
</property>
</bean>
您也可以使用nested properties更简洁地做到这一点:
<bean id="supedUpMiniVan" parent="myWifesCar">
<property name="engine.volume" value="600"/>
<property name="engine.weight" value="750"/>
</bean>
这将使用“defaultEngine”.但是,如果您以这种方式创建两个车,每个车辆具有不同的属性值,行为将不正确.这是因为两辆汽车将共享相同的引擎实例,第二辆汽车覆盖了第一辆车上设置的属性设置.这可以通过将defaultEngine标记为“原型”来进行补救,每次引用时都会实例化一个新引擎:
<bean id="defaultEngine" class="Engine" scope="prototype">
<property name="numberOfCylinders" value="4"/>
<property name="volume" value="400"/>
<property name="weight" value="475"/>
</bean>
我想这个例子给出了基本的想法.如果您的数据结构很复杂,您可以定义多个抽象Bean,或者创建几个不同的抽象层次结构,尤其是如果您的bean层次结构比两个bean更深.
旁注:我的示例使用属性,我相信在Spring xml和Java代码中都更清楚了解.然而,完全相同的技术适用于构造函数,工厂方法等.