spring嵌套事务导致rollback-only

本文详细分析了Spring中事务管理出现UnexpectedRollbackException的原因,主要涉及事务的传播行为和嵌套事务。当PROPAGATION_REQUIRED和PROPAGATION_NESTED等传播行为混淆使用时,可能导致事务回滚异常。解决方案包括正确处理异常和调整事务传播属性,以确保事务的正确提交和回滚。同时,文中列举了Spring的事务传播性定义,并对比了PROPAGATION_REQUIRES_NEW和PROPAGATION_NESTED的区别。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

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

背景及分析原因:

  1. 进行保存操作并开启事务 T1
    1.1 进行保存A
    1.2 进行保存B 采用mybatisplus的批量保存(注意批量保存会开启事务T2 因为默认传播行为Propagation.REQUIRED所以把T2加入T1) 保存失败则 被调用方 trycatch 捕获 不向外抛出异常(这里会把T1的rollback-only修改为true 因为t2加入t1)
    1.3 进行保存C
  2. 最终完成保存,T1没有抛出异常 T2抛出异常(需要回滚)被方法捕获住,一个需要回滚,一个需要提交,所以导致rollback-only异常

示例代码:

Class ServiceA {
    @Resource(name = "serviceB")
    private ServiceB b;
    
    @Transactional
    public void a() {
        try {
            b.b()
        } catch (Exception ignore) {
        }
    }
}

Class ServiceB {
    @Transactional
    public void b() {
        throw new RuntimeException();
    }
}

异常抛出方法:org.springframework.transaction.support.AbstractPlatformTransactionManager#processRollback

解决方案

  1. 如果希望内层事务抛出异常时中断程序执行,直接在外层事务的catch代码块中抛出e.
  2. 如果希望程序正常执行完毕,并且希望外层事务结束时全部提交,需要在内层事务中做异常捕获处理。
  3. 如果希望内层事务回滚,但不影响外层事务提交,需要将内层事务的传播方式指定为PROPAGATION_NESTED。注:PROPAGATION_NESTED基于数据库savepoint实现的嵌套事务,外层事务的提交和回滚能够控制嵌内层事务,而内层事务报错时,可以返回原始savepoint,外层事务可以继续提交。

Spring中对于事务传播性的几种定义

  1. PROPAGATION_REQUIRED – 支持当前事务,如果当前没有事务,就新建一个事务。这是最常见的选择。
  2. PROPAGATION_SUPPORTS – 支持当前事务,如果当前没有事务,就以非事务方式执行。
  3. PROPAGATION_MANDATORY – 支持当前事务,如果当前没有事务,就抛出异常。
  4. PROPAGATION_REQUIRES_NEW – 新建事务,如果当前存在事务,把当前事务挂起。
  5. PROPAGATION_NOT_SUPPORTED – 以非事务方式执行操作,如果当前存在事务,就把当前事务挂起。
  6. PROPAGATION_NEVER – 以非事务方式执行,如果当前存在事务,则抛出异常。
  7. PROPAGATION_NESTED – 如果当前存在事务,则在嵌套事务内执行。如果当前没有事务,则进行与PROPAGATION_REQUIRED类似的操作。

前六个策略类似于EJB CMT,第七个(PROPAGATION_NESTED)是Spring所提供的一个特殊变量。
它要求事务管理器或者使用JDBC 3.0 Savepoint API提供嵌套事务行为(如Spring的DataSourceTransactionManager)
其中对于PROPAGATION_REQUIRES_NEW与PROPAGATION_NESTED的理解上有些类似,关键在于嵌套事务的理解。
来看一下网上大神对Juergen Hoeller表述的翻译:

PROPAGATION_REQUIRES_NEW 启动一个新的, 不依赖于环境的 “内部” 事务. 这个事务将被完全 commited 或 rolled back 而不依赖于外部事务, 它拥有自己的隔离范围, 自己的锁, 等等. 当内部事务开始执行时, 外部事务将被挂起, 内务事务结束时, 外部事务将继续执行.
另一方面, PROPAGATION_NESTED 开始一个 “嵌套的” 事务, 它是已经存在事务的一个真正的子事务. 潜套事务开始执行时, 它将取得一个 savepoint. 如果这个嵌套事务失败, 我们将回滚到此 savepoint. 潜套事务是外部事务的一部分, 只有外部事务结束后它才会被提交.

在这里插入图片描述

参考文章:https://yunlongn.github.io/2019/05/06/%E8%AE%B0%E4%B8%80%E6%AC%A1%E4%BA%8B%E5%8A%A1%E7%9A%84%E5%9D%91Transaction-rolled-back-because-it-has-been-marked-as-rollback-only/
参考文章:https://blog.youkuaiyun.com/f641385712/article/details/80445912
参考文章:https://blog.youkuaiyun.com/yanxin1213/article/details/100582643

出现错误信息 `"Transaction rolled back because it has been marked as rollback-only"` 通常出现在使用 **Spring 框架进行事务管理** 的场景中,尤其是在 **声明式事务(@Transactional)** 中。 --- ### 🧠 错误含义解释: 当一个事务被标记为 `rollback-only` 后,即使后续代码试图正常提交事务Spring 也会强制回滚该事务,并抛出如下异常: ``` org.springframework.transaction.UnexpectedRollbackException: Transaction rolled back because it has been marked as rollback-only ``` 这意味着: > 虽然你没有主动抛出异常或调用回滚,但事务已经被某个环节标记为“只能回滚”,最终 Spring 在提交时发现这个标记,就会抛出此异常。 --- ## ✅ 常见原因分析 ### 1. **子方法抛出异常导致事务回滚** 如果你在事务方法中调用了另一个带有 `@Transactional(propagation = Propagation.REQUIRES_NEW)` 的方法,并且该方法抛出了未处理的异常,那么外层事务也会被标记为 rollback-only。 ```java @Transactional public void outerMethod() { innerService.innerMethod(); // 抛出异常 -> 标记为 rollback-only } @Service class InnerService { @Transactional(propagation = Propagation.REQUIRES_NEW) public void innerMethod() { throw new RuntimeException("出错了"); } } ``` ### 2. **手动设置了回滚** 你在代码中显式调用了: ```java TransactionAspectSupport.currentTransactionStatus().setRollbackOnly(); ``` 这会直接标记当前事务rollback-only。 ### 3. **事务传播行为不当** 比如你使用了 `Propagation.NESTED` 或 `REQUIRES_NEW`,并且嵌套事务失败,而你没有正确地捕获和处理异常。 ### 4. **全局事务配置问题(如分布式事务)** 如果你使用的是 JTA 或者 Seata 等分布式事务框架,某些分支事务失败也可能导致整个事务被标记为 rollback-only--- ## ✅ 如何定位与解决? ### ✅ 查看完整的堆栈跟踪日志 查看完整异常堆栈,找到最早触发事务回滚的地方。通常是某个子事务或内部方法抛出了异常。 ### ✅ 示例修复方案 #### 情况一:子方法异常导致事务被标记为回滚 你可以捕获异常并防止事务传播失败: ```java @Transactional public void outerMethod() { try { innerService.innerMethod(); } catch (Exception e) { // 处理异常,避免事务继续传播失败 log.error("Inner method failed", e); } } ``` #### 情况二:设置回滚策略 如果你希望某些异常不触发回滚,可以配置 `@Transactional(noRollbackFor = SomeException.class)` ```java @Transactional(noRollbackFor = MyCustomException.class) public void someMethod() { // 即使抛出 MyCustomException,也不会回滚 } ``` --- ## ✅ 最佳实践建议 | 场景 | 建议 | |------|------| | 避免事务嵌套过深 | 使用清晰的事务边界控制 | | 异常处理要明确 | 捕获并决定是否回滚 | | 日志追踪 | 打印事务状态和回滚原因 | | 不要用 try-catch 忽略所有异常 | 可能掩盖真正的错误 | ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值