Transaction rolled back because it has been marked as rollback-only 问题记录

在运行项目时遇到Spring事务异常'UnexpectedRollbackException',问题源于MethodB的数据异常导致事务被标记为rollback-only,而MethodA通过try-catch消化异常未继续抛出。查看源码后,解决方案包括将A方法的隔离级别改为'SUPPORTS',或者在Controller层处理异常。Spring事务默认仅对RuntimeException及其子类进行回滚。

今天高高兴兴上班,一打开电脑运行项目,突然就提示

org.springframework.transaction.UnexpectedRollbackException: Transaction rolled back because it has been marked as rollback-only

啥,什么操作,代码也没动呀,🤔那么,开始认真的分析,首先,然后 MethodA()-> MethodB(),MethodB 数据异常引起了业务异常,B 方法向上抛出了异常,在B()返回的时候,transaction被设置为rollback-only了,但是A()正常消化掉 try-catch 掉了,没有继续向外抛。那么A()结束的时候,transaction会执commit操作,但是transaction 中已经被设置为rollback-only了。

所以会出现这个错误。

看下源码

public final void commit(TransactionStatus status) throws TransactionException {
        if (status.isCompleted()) {
            throw new IllegalTransactionStateException("Transaction is already completed - do not call commit or rollback more than once per transaction");
        } else {
            DefaultTransactionStatus defStatus = (DefaultTransactionStatus)status;
            if (defStatus.isLocalRollbackOnly()) {
                if (defStatus.isDebug()) {
                    this.logger.debug("Transactional code has requested rollback");
                }

                this.processRollback(defStatus);
            //会执行该判断
            } else if (!this.shouldCommitOnGlobalRollbackOnly() && defStatus.isGlobalRollbackOnly()) {
               
                if (defStatus.isDebug()) {
                    this.logger.debug("Global transaction is marked as rollback-only but transactional code requested commit");
                }
                 //执行回滚
                this.processRollback(defStatus);
                // 异常判定,也是本问题抛出异常的位子
                if (status.isNewTransaction() || this.isFailEarlyOnGlobalRollbackOnly()) {
                    throw new UnexpectedRollbackException("Transaction rolled back because it has been marked as rollback-only");
                }
            } else {
                this.processCommit(defStatus);
            }
        }

那么该如何解决呢?

因为我的 A()是一个只读方法,所以我重新配置了隔离机制 SUPPORTS

@Transactional(readOnly = true,propagation = Propagation.SUPPORTS)

这个状态用一句话概括就是“有则加入事物,无也不创建事物”。对于只读的方法来说,其实不启用事务,也是没有关系的。

就跳过了在A() 结束Trasaction commit 的操作,也就避免了异常的抛出,当然还可以改成继续向上排除异常,在Controller 层处理。

注:自己手动抛出的异常,尽可能是RuntimeException 或 其子类,因为Spring的事务默认回滚是发生RuntimeException,

即使配置了rollbackFor = Exception.class也不管用

回顾以下Spring的事务隔离级别:

MANDATORY 
          Support a current transaction, throw an exception if none exists.
NESTED 
          Execute within a nested transaction if a current transaction exists, behave like PROPAGATION_REQUIRED else.
NEVER 
          Execute non-transactionally, throw an exception if a transaction exists.
NOT_SUPPORTED 
          Execute non-transactionally, suspend the current transaction if one exists.
REQUIRED 
          Support a current transaction, create a new one if none exists.
REQUIRES_NEW 
          Create a new transaction, suspend the current transaction if one exists.
SUPPORTS 
          Support a current transaction, execute non-transactionally if none exists.

 

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值