简洁的代理定义
特别是在定义事务性代理(transactional proxies)时,你可能会得到许多相似的代理定义。使用父 bean(parent bean)和子 bean(child bean)的定义,再加上内部 bean(inner bean)的定义,可以使得代理定义更加清晰简洁。
首先,我们创建一个父类、模板以及代理的Bean定义,如下所示:
<bean id="txProxyTemplate" abstract="true"
class="org.springframework.transaction.interceptor.TransactionProxyFactoryBean">
<property name="transactionManager" ref="transactionManager"/>
<property name="transactionAttributes">
<props>
<prop key="*">PROPAGATION_REQUIRED</prop>
</props>
</property>
</bean>
这个(代理)本身从未被实例化过,因此它实际上可能是不完整的。然后,每个需要创建的代理都是一个子bean定义,该子bean定义将代理的目标对象作为内部bean定义来封装,因为目标对象本身也从来不会被单独使用。以下示例展示了一个这样的子bean:
<bean id="myService" parent="txProxyTemplate">
<property name="target">
<bean class="org.springframework.samples.MyServiceImpl">
</bean>
</property>
</bean>
你可以覆盖父模板中的属性。在以下示例中,我们覆盖了事务传播设置:
<bean id="mySpecialService" parent="txProxyTemplate">
<property name="target">
<bean class="org.springframework.samples.MySpecialServiceImpl">
</bean>
</property>
<property name="transactionAttributes">
<props>
<prop key="get*">PROPAGATION_REQUIRED,readOnly</prop>
<prop key="find*">PROPAGATION_REQUIRED,readOnly</prop>
<prop key="load*">PROPAGATION_REQUIRED,readOnly</prop>
<prop key="store*">PROPAGATION_REQUIRED</prop>
</props>
</property>
</bean>
请注意,在父Bean示例中,我们通过将abstract属性设置为true来明确标记父Bean定义为抽象类,如之前所描述的那样,这样它实际上可能永远不会被实例化。应用程序上下文(但简单的Bean工厂除外)默认会预先实例化所有单例对象。因此,至少对于单例Bean来说,如果你有一个仅打算用作模板的(父)Bean定义,并且该定义指定了一个类,那么你必须确保将abstract属性设置为true。否则,应用程序上下文实际上会尝试预先实例化它。