当前位置: 首页 > news >正文

Spring Boot事务失效场景及解决方案

事务失效场景1:方法非public修饰

原因
Spring事务基于动态代理(AOP)实现,非public方法无法被代理拦截,导致事务失效。

代码示例

@Service
public class OrderService {@Transactionalprivate void createOrder() { // 非public方法// 业务逻辑}
}

解决方案

  • 将方法改为public修饰。
  • 若需限制方法访问权限,可通过编程式事务(TransactionTemplate)实现。

事务失效场景2:自调用问题

原因
同类中方法A调用方法B(带@Transactional),由于代理机制失效,事务不生效。

代码示例

@Service
public class UserService {public void updateUser() {this.saveUser(); // 自调用}@Transactionalpublic void saveUser() {// 数据库操作}
}

解决方案

  • 将事务方法拆分到另一个类中,通过注入调用。
  • 使用AopContext.currentProxy()获取代理对象(需开启exposeProxy)。

事务失效场景3:异常类型未被捕获

原因
默认仅对RuntimeExceptionError回滚,若抛出其他异常(如IOException)且未配置rollbackFor,事务不会回滚。

代码示例

@Transactional
public void processData() throws IOException {// 抛出IOExceptionthrow new IOException("文件异常");
}

解决方案

  • 明确指定回滚异常类型:
    @Transactional(rollbackFor = Exception.class)
    

事务失效场景4:事务传播行为配置错误

原因
例如REQUIRES_NEW嵌套使用时,内层事务失败可能不影响外层事务。

代码示例

@Transactional(propagation = Propagation.REQUIRED)
public void outerMethod() {innerMethod(); // 内层事务独立提交
}@Transactional(propagation = Propagation.REQUIRES_NEW)
public void innerMethod() {// 操作失败但outerMethod继续执行
}

解决方案

  • 根据业务需求调整传播行为,如改为REQUIRED
  • 避免过度嵌套事务。

事务失效场景5:多数据源未指定事务管理器

原因
多数据源环境下未明确指定transactionManager,导致事务绑定到默认管理器。

代码示例

@Transactional // 默认使用primary事务管理器
public void saveToSecondaryDB() {// 操作secondary数据源
}

解决方案

  • 注解中指定事务管理器:
    @Transactional("secondaryTransactionManager")
    

事务失效场景6:手动捕获异常未抛出

原因
捕获异常后未重新抛出,事务拦截器无法触发回滚。

代码示例

@Transactional
public void updateOrder() {try {// 数据库操作} catch (Exception e) {log.error("错误", e); // 未抛出异常}
}

解决方案

  • catch块中抛出RuntimeException
  • 或使用TransactionAspectSupport.currentTransactionStatus().setRollbackOnly()手动回滚。

事务失效场景7:非事务方法调用事务方法

原因
若父类方法未开启事务,调用子类@Transactional方法时,代理失效。

代码示例

public class BaseService {public void execute() {save(); // 事务失效}@Transactionalpublic void save() {}
}

解决方案

  • 将事务注解添加到父类方法。
  • 避免通过继承层级调用事务方法。

总结
  • 检查方法修饰符和代理机制。
  • 确保异常类型和传播行为匹配业务需求。
  • 多数据源需显式指定事务管理器。
  • 优先通过设计规避自调用问题。
http://www.xdnf.cn/news/682057.html

相关文章:

  • 自动驾驶决策规划框架详解:从理论到实践
  • 【C语言干货】回调函数
  • endnote2025安装教程以及激活文件
  • ELectron 中 BrowserView 如何进行实时定位和尺寸调整
  • Asp.Net Core 如何配置在Swagger中带JWT报文头
  • leetcode hot100刷题日记——21.不同路径
  • 六、西方哲学
  • 【连载19】基础智能体的进展与挑战综述-对智能体大脑的威胁
  • halcon高斯滤波
  • 网络编程--上篇
  • 【详细记录】我的第一次裸片硬件尝试:stm32f103c8t6最小核心板一板成
  • unet 视频截图 实现简单的unet kaggle运行
  • Kruskal-Wallis检验 vs. 多次Wilcoxon检验:多重比较-spss
  • LCR 094. 分割回文串 II
  • Elasticsearch搜索机制与分页优化策略
  • Pytest自动化测试框架搭建:Jenkins持续集成
  • 通俗解释网络参数RTT(往返时间)
  • Scratch节日 | 六一儿童节
  • 并发编程(二)—synchronized和volatile
  • 尚硅谷redis7 55-57 redis主从复制之理论简介
  • 从零搭建上门做饭平台:高并发订单系统设计
  • 普罗米修斯监控CPU\内存汇聚图
  • 产业集群间的专利合作关系
  • Visual Studio编译当前文件
  • vue项目 build时@vue-office/docx报错
  • ceph recovery 相关参数
  • MMdetection推理验证输出详解(单张图片demo)
  • 用DEEPSEEK写的扫雷小游戏
  • 如何设计高效的索引策略?
  • 一则doris数据不一致问题