毛线颜色拼图
70.92MB · 2025-12-04
在 Spring 的事务管理中,REQUIRED 是最常用也是默认的事务传播属性。很多开发者在使用时会遇到一个常见的困惑:为什么内部方法抛出的异常即便被外层捕获了,事务还是会整体回滚? 本文将深入剖析 REQUIRED 下的事务回滚机制及其背后的原理。
这意味着调用链上的所有方法都运行在同一个物理事务里,彼此之间没有隔离。只要其中一个方法触发回滚条件,整个事务都会被撤销。
Spring 默认回滚规则:
RuntimeException)和错误(Error):触发回滚。CheckedException):不会触发回滚,除非在 @Transactional(rollbackFor = Exception.class) 中显式指定。假设调用链如下:
@Transactional(propagation = Propagation.REQUIRED)
public void outer() {
try {
inner();
} catch (Exception e) {
// 异常被捕获
}
}
@Transactional(propagation = Propagation.REQUIRED)
public void inner() {
throw new RuntimeException("内部错误");
}
执行过程:
outer() 启动时创建事务。inner() 加入同一事务并抛出异常。outer() 捕获了异常,rollback-only 标记不会自动清除。rollback() 而不是 commit()。因此,捕获异常并不能阻止事务回滚。
原因在于 Spring 的事务是基于 事务状态标记 而不是你的 try-catch 来决定的:
TransactionInterceptor 判断是否需要回滚。status.setRollbackOnly()。最终由事务管理器在 commit() 阶段检查 rollback-only 标记,决定整体回滚。
如果确实需要内部异常不影响外部事务,可以考虑以下方式:
@Transactional(noRollbackFor = RuntimeException.class)
public void inner() {
throw new RuntimeException("不会触发回滚");
}
但这种方式风险极高,可能提交错误数据。
catch (Exception e) {
TransactionAspectSupport.currentTransactionStatus().setRollbackOnly(false);
}
此方式非常危险,通常不建议。
REQUIRES_NEW:为内部方法开启新事务,失败只影响内部,不影响外部。NESTED:使用保存点,内部失败可回滚到保存点,外部事务继续。这两种方式才是更优雅的解决方案。
核心逻辑在 TransactionAspectSupport.invokeWithinTransaction():
commit() 还是 rollback()。REQUIRED 下,所有方法共享一个事务。REQUIRES_NEW 或 NESTED。一句话总结:在 Spring 的 REQUIRED 模式下,事务是一荣俱荣、一损俱损的整体,内部异常即便被捕获也会让整个事务回滚,真正想做到“局部回滚”需要换传播属性来实现。