一、何时失效
Spring事务是AOP的应用,扫描@Transactional注解,调用事务处理器的begin(),commit(),rollack()等方法。
1、@Transactional 应用在非 public 方法上
spring 事务底层在扫描 @Transactional 注解时会直接不会获取非 public 的方法上的 @Transactional 注解的属性信息,使用JDK或者CGLIB代理时不会增强private方法。
2、@Transactional 的 rollbackFor 设置
默认不设置的话,在 Error 和 RunTimeException 错误时会触发事务,其他异常不会,通常工程里都会添加rollbackFor = Exception.class,拦截受检查异常;rollbackFor 指定异常类型,则只会在该异常类型出现时触发事务;
3、同一个类中的方法调用
同一个类中,methodA 没有设置事务,methodB 设置了事务,methodA 调用 methodB 时,事务失效(AOP invoke时只增强了A方法,但如果B在另一个Bean中,调用methodB也会得到增强,事务生效);只有当事务方法被当前类以外的代码调用时,才会由 spring 生成代理对象管理事务;
4、异常被 catch 住了,不会再触发事务
在业务代码中一般不需要 catch 异常,如果非要 catch 一定要抛出 throw new RunTimeException() ,或注解中指定异常类型
@Transactional(rollbackFor=Exception.class),否则会导致事务失效。
5、数据库存储引擎不支持事务,MYISAM等
二、事务的传播行为
事务传播行为用来描述由某一个事务传播行为修饰的方法被嵌套进另一个方法的时事务如何传播。
用伪代码说明:
public void methodA(){
methodB();
//doSomething
}
@Transaction(Propagation=XXX)
public void methodB(){
//doSomething
}
代码中methodA()
方法嵌套调用了methodB()
方法,methodB()
的事务传播行为由@Transaction(Propagation=XXX)
设置决定。这里需要注意的是methodA()
并没有开启事务,某一个事务传播行为修饰的方法并不是必须要在开启事务的外围方法中调用。
事务传播行为类型 | 说明 |
---|---|
PROPAGATION_REQUIRED | 如果当前没有事务,就新建一个事务,如果已经存在一个事务中,加入到这个事务中。这是最常见的选择。 |
PROPAGATION_SUPPORTS | 支持当前事务,如果当前没有事务,就以非事务方式执行。 |
PROPAGATION_MANDATORY | 使用当前的事务,如果当前没有事务,就抛出异常。 |
PROPAGATION_REQUIRES_NEW | 新建事务,如果当前存在事务,把当前事务挂起。 |
PROPAGATION_NOT_SUPPORTED | 以非事务方式执行操作,如果当前存在事务,就把当前事务挂起。 |
PROPAGATION_NEVER | 以非事务方式执行,如果当前存在事务,则抛出异常。 |
PROPAGATION_NESTED | 如果当前存在事务,则在嵌套事务内执行。如果当前没有事务,则执行与PROPAGATION_REQUIRED类似的操作。 |