spring通过springAOP提供事务支持。可理解spring事务将方法包装上事务边界的切面。
spring提供三种方式来声明事务性边界。以前通过SpringAOP和TransactionProxyFactoryBean的代理Bean实现声明式事务。后来更好的声明式事务是使用spring的
tx命名空间和
@Transactional注解。
spring提供了多种声明式事务的机制,但所有的方式都依赖五中参数控制管理事务策略。
事务的属性
传播行为 定义了客户端与被调用方法之间的事务边界
传播行为常量
隔离级别 定义了一个事务可能受其他并发事务影响的程度
并发可能会导致出现几种问题:
隔离级别:
是否只读
若事务被声明为只读,则数据库会根据事务进行一些优化。
对那些具备启动一个新事务的传播行为(PROPAGATION_REQUIRED、PROPAGATION_REQUIRED_NEW、PROPAGATION_NESTED)的方法来说,将事务声明为只读才有意义。
另外,如果采用hibernate作为持久化机制,那么将事务声明为只读将会导致hibernate的flush模式将被设置为FLUSH_NEVER。这会告诉hibernate避免和数据库进行不必要的对象同步,并将所有的更新延迟到事务结束。
事务超时
为了使应用程序很好地运行,事务不能运行太长的时间。因为如果事务的运行时间变得特别长,而事务可能涉及到对后端数据库的锁定,所以长时间的事务会不必要地占用数据库资源。那么你可以声明一个事务,在特定的秒数后自动回滚,而不是等待结束。
因为超时时钟会在事务开始启动。所以只有对那些具备可能启动一个新事务的传播行为(PROPAGATION_REQUIRED、PROPAGATION_NEW、PROPAGATION_NESTED)的方法来说,声明事务超市才有意义。
回滚规则
默认情况下事务遇到运行期异常时才会回滚。而遇到检查型异常时不会回滚。
在xml中定义事务
使用
tx的命名空间,可以避免TransactionProxyFactoryBean冗长的配置文件。
在xml中声明事务策略
<tx:advice id="txAdvice">
<tx:attributes>
<tx:method name="save" propagation="REQUIRED" />
</tx:method name="*" propagation="SUPPORTS" read-only="true" />
<tx:attributes>
</tx:advice>
<tx:method>有多个属性来帮助定义方法的事务策略
根据约定优于配置,<tx:advice>假定事务管理器被声明为一个id为transactionManager的bean。如果碰巧为事务管理器配置一个不同的id(如txManager),则需要在transactionManager属性中明确事务管理器的id:
<tx:advice id="txAdvice" transaction-Manager="txManager">
</tx:advice>
以上的<tx:advice>只是定义了AOP通知,用于把事务边界通知给方法。但这不是完整事务性切面。为了定义完整事务性切面,我们需要一个通知器(advisor)。如下:
<aop:config>
<aop:advisor pointcut="execution(** ..SpitterService.*.(..))" advice-ref="txAdvice" />
</aop:config>
定义注解驱动的事务
tx命名空间还提供了<tx:annotation-driven>元素。使用十分简单:
<tx:annotation-driven />
<tx:annotation-driven>告诉spring检查上下文中所有的bean并查找使用@Transactional注解的bean。
对每个使用@Transactional注解的bean,<tx:annotation-driven>会自动为它添加事务。通知的事务属性是通过@Transactional注解的参数来定义的。
参考资料:《spring 实战(第三版)》
图片来自:《spring 实战(第三版)》