Spring 事务管理 Transaction rolled back because it has been marked as rollback-only

举个例子:

假设我们有一个服务类的方法A,它调用了方法B。方法B抛出了一个异常,但是方法A捕获了这个异常并没有重新抛出。但是,方法B的事务传播行为可能是REQUIRED,所以方法B会在同一个事务中执行。当方法B抛出异常时,事务被标记为回滚。然后方法A捕获了异常,没有重新抛出,那么方法A的事务拦截器在退出时会尝试提交事务,但是发现事务已经被标记为回滚,于是就会回滚事务并输出上述信息。

解决:

检查代码中是否有在事务方法中捕获了异常但没有正确处理的情况。如果希望事务回滚,通常应该在捕获异常后,将异常重新抛出(或者不再捕获异常,让异常自动抛出)。

如果确实需要捕获异常并且不想回滚事务,那么可以考虑将可能抛出异常的方法的事务传播行为设置为REQUIRES_NEW,这样它会在一个新的事务中执行,不会影响当前事务。

但是,请注意,REQUIRES_NEW会开启一个新事务,如果原事务已经存在,则会挂起原事务,这样方法B的异常就不会导致原事务回滚。

另一种情况是,可能你并不想在一个事务中运行整个方法链,那么可以考虑调整事务的边界,将不需要事务的方法排除,或者将事务的传播行为设置为NOT_SUPPORTED等。

典型代码场景

java 复制代码
@Service
public class UserService {
    
    @Transactional
    public void outerMethod() {
        try {
            innerMethod();  // 内层方法抛出异常
            // 其他业务逻辑...
        } catch (Exception e) {
            // 捕获异常,希望继续执行
            log.error("发生错误", e);
        }
        // 这里Spring会尝试提交事务,但事务已被标记为rollback-only
    }
    
    @Transactional(propagation = Propagation.REQUIRED)
    public void innerMethod() {
        // 业务操作...
        throw new RuntimeException("业务异常"); // 导致事务被标记为rollback-only
    }
}

解决方案

方案1:重新抛出异常(推荐)

java 复制代码
@Transactional
public void outerMethod() {
    try {
        innerMethod();
    } catch (Exception e) {
        log.error("发生错误", e);
        throw e; // 重新抛出异常,让事务正常回滚
    }
}

方案2:使用不同的事务传播机制

java 复制代码
@Service
public class UserService {
    
    @Transactional
    public void outerMethod() {
        try {
            // 使用REQUIRES_NEW创建新事务
            innerMethodWithNewTransaction();
        } catch (Exception e) {
            log.error("内层方法失败,但外层事务继续", e);
        }
        // 外层事务可以正常提交
    }
    
    @Transactional(propagation = Propagation.REQUIRES_NEW)
    public void innerMethodWithNewTransaction() {
        // 这个方法的异常不会影响外层事务
        throw new RuntimeException("业务异常");
    }
}

方案3:手动控制事务边界

java 复制代码
@Service
public class UserService {
    
    @Autowired
    private TransactionTemplate transactionTemplate;
    
    public void outerMethod() {
        // 手动控制事务
        transactionTemplate.execute(status -> {
            try {
                innerMethod();
                return true;
            } catch (Exception e) {
                status.setRollbackOnly(); // 明确标记回滚
                log.error("事务回滚", e);
                return false;
            }
        });
    }
}

方案4:使用声明式事务控制

java 复制代码
@Service
public class UserService {
    
    @Transactional
    public void outerMethod() {
        innerMethod();
        // 让异常自然传播,不要捕获
    }
    
    // 指定回滚的异常类型
    @Transactional(rollbackFor = Exception.class)
    public void innerMethod() {
        // 业务逻辑
    }
}

预防措施

  1. 统一异常处理策略:在事务边界明确异常处理方式
  2. 合理使用传播机制
    • REQUIRED(默认):加入现有事务
    • REQUIRES_NEW:创建新事务
    • NESTED:嵌套事务
  3. 避免在事务方法中静默捕获异常
  4. 使用事务监听器:监控事务状态
java 复制代码
// 事务事件监听
@Component
public class TransactionEventListener {
    
    @TransactionalEventListener(phase = TransactionPhase.AFTER_ROLLBACK)
    public void handleRollback(TransactionRolledBackEvent event) {
        log.warn("事务回滚: {}", event.getTransactionName());
    }
}
相关推荐
helloworldandy10 小时前
使用Pandas进行数据分析:从数据清洗到可视化
jvm·数据库·python
数据知道11 小时前
PostgreSQL 故障排查:如何找出数据库中最耗时的 SQL 语句
数据库·sql·postgresql
qq_124987075312 小时前
基于SSM的动物保护系统的设计与实现(源码+论文+部署+安装)
java·数据库·spring boot·毕业设计·ssm·计算机毕业设计
枷锁—sha12 小时前
【SRC】SQL注入WAF 绕过应对策略(二)
网络·数据库·python·sql·安全·网络安全
Coder_Boy_12 小时前
基于SpringAI的在线考试系统-考试系统开发流程案例
java·数据库·人工智能·spring boot·后端
Gain_chance12 小时前
35-学习笔记尚硅谷数仓搭建-DWS层最近n日汇总表及历史至今汇总表建表语句
数据库·数据仓库·hive·笔记·学习
2301_8187320612 小时前
前端调用控制层接口,进不去,报错415,类型不匹配
java·spring boot·spring·tomcat·intellij-idea
此生只爱蛋12 小时前
【Redis】主从复制
数据库·redis
码字的字节12 小时前
Spring Cloud服务注册与发现(一):手把手搭建Eureka Server,详解高可用配置
spring·spring cloud·eureka
大厂资深架构师12 小时前
Spring Cloud Eureka在后端系统中的服务剔除策略
spring·spring cloud·ai·eureka