@Transation在什么情况下会失效

Spring的事务管理机制分为两种:编程式事务(代码中手动开启事务)和声明式事务(使用注解开启事务)

编程式事务:

代码中手动管理事务的回滚、提交,代码侵入性比较强。

try { //TODO something transactionManager.commit(status); } catch (Exception e) { transactionManager.rollback(status); throw new InvoiceApplyException("异常失败"); }

声明式事务:

基于AOP切面的,它将具体业务代码和事务处理代码部分解耦,代码侵入性很低(使用cglib动态代理),所以实际开发中用声明式事务用的比较多,声明式事务有两种实现方式:

1.基于TX和AOP的xml配置文件方式

2.基于Transsactional注解

@Trancational介绍

作用位置:接口、类、类方法

1.作用于接口:(不推荐)

在接口上声明后,而Spring AOP配置的代理方式又是Cglib代理的话,会造成@Transational失效。

2.作用于类:

类中的所有public方法都会配置成相同的事务属性信息。

3.作用于类方法:

当类与类方法都配置了@Transational属性的话,方法上的事务会覆盖类上的方法。

@Trancational属性

propagation属性:(代表事务的传播行为,默认值为Propagation.REQUIRED)

》Propagation.REQUIRED:如果当前存在事务则加入该事务,如果不存在事务,则创建一个新的事务。(也就是说A方法和B方法都添加了注解,在默认的REQUIRED属性下,A方法内部调用B方法,并把B方法加入到A方法中,合并为一个事务)

》Propagation.SUPPORTS:如果当前存在事务,则加入该事务;如果当前不存在事务,则以非事务的方式继续运行。

》Propagation.MANDATORY:如果当前存在事务,则加入事务;如果当前不存在事务,则抛出异常

》Propagation.REQUIRES_NEW:重新创建一个新的事务,如果存在事务,暂停当前的事务。(当类中A中的a方法用默认Propagation.REQQUIRED模式,类B中的b方法采用Propagation.REQQUIRED_NEW模式,然后在a方法中调用b方法操作数据库,然而a方法抛出异常,b方法正常执行,则b不进行回滚,因为Propagation.REQUIRED_NEW会暂停a方法)

》Propagation.NOT_SUPPORTED:以非事务的方式运行,如果当前存在事务,暂停当前的事务。

》Propagation.NEVER:以非事务的方式运行,如果存在事务,则抛出异常

》Propagation.NESTED:和Propagation.REQUIRED效果一样

isolation属性:事务的隔离级别,默认值为Isolation.DEFFAULT

》Isolation.DEFAULT:使用底层数据库默认的隔离级别

》Isolation.READ_UNCOMMITED

》Isolation.READ_COMMITTED

》Isolation.REPEATABLE_READ

》Isolation.SERIALIZABLE

timeout属性:

事务的超时时间,默认值为 -1。如果超过该时间限制但事务还没有完成,则自动回滚事务。

readOnly:指定事务是否为只读事务,默认值为 false;为了忽略那些不需要事务的方法,比如读取数据,可以设置 read-only 为 true。

rollbackFor 属性

rollbackFor :用于指定能够触发事务回滚的异常类型,可以指定多个异常类型。

noRollbackFor属性

noRollbackFor:抛出指定的异常类型,不回滚事务,也可以指定多个异常类型。

@Transational失效场景

1.@Transational作用于非public方法上

AOP代理时,会通过事务拦截器(TransactionInterceptor)在目标方法执行前后进行拦截,代理类中的生效方法会间接调用computeTransactionAttribute方法读取事务注解的配置信息

,会检查该注解的目标方法是否为public,不是则不会增强事务处理信息。也就是代理类并未加强非public的方法

2.@Transational使用的Papagation传播行为不对

如使用了SUPPORT、NOT_SUPPORT、NERVER等属性需要注意

3.@Transational使用的rollbackFor设置错误

可以指定事务回滚的异常类型,Spring默认的抛出的是未检查的异常,继承于RuntimeExcetion或者Error的类(子类会回滚,Spring内部会使用getDepth递归查找子类),其他异常不回滚。如果希望Spring处理其他类型的异常,可以设置rollbackFor属性。

4.一个没有事务注解的方法调用有事务注解的方法,导致@Transational失效

因为AOP的代理造成的,没有事务注解的方法,方法没有增强,且在当前方法中调用有事务注解的方法时,调用的为当前类对象的方法,并不是代理类的增强方法。

5.异常被catch吃掉了

有事务注解的业务A类a方法调用有事务注解B类b方法,b方法抛出异常,a方法catch(事务注解默认属性)

假如业务B中的b方法有事务注解且抛出异常,业务B类认为要回滚,而a方法catch捕获处理,则ServiceA类任务应该正常提交。就会造成前后不一致。

spring的事务在方法之前开启的,事务回滚还是提交,取决于代码是否抛出Runtime异常,或者catch到后是否抛出Runtime异常,如果没有的话,事务不会回滚。

业务方法一般不需要try catch,如果非要catch记得抛出Runtime异常,或者rollback属性设置为Exception.class;

6.数据库引擎不支持事务

 

  • 2
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值