从@Transactional失效场景到传播行为原理

大家有没有见过这种场景:就是再写后端接口的时候,你的实现层上面写了事务回滚的注解,但是前端调用接口传参插入或修改时即使报错了,但是这些垃圾数据还是添加或者修改了数据库中的数据呢!

难道注解失效了吗?但一旦遇到这种问题就容易懵圈。 那么这次就好好的分析分析从常见的@Transactional失效场景到传播行为的原理的理解

一、@Transactional为什么会失效?

先来看一个实际开发中经常遇到的场景:你在一个类的方法上加了@Transactional注解,满心期待着出现异常时数据会自动回滚,结果却发现事务根本没生效!这种情况通常有以下几种原因:

  1. 注解放在了私有方法上 :Spring事务是通过代理实现的,私有方法无法被代理,所以放在私有方法上的@Transactional注解根本不会生效。记住,它只能放在public方法上。

  2. 自调用问题:这是最容易出现问题的。当一个类中的方法A(没有事务注解)调用同一个类中的方法B(有@Transactional注解)时,事务是不会生效的。因为Spring的事务管理是通过代理对象实现的,自调用绕过了代理机制。

java 复制代码
@Service
public class UserService {
    
    public void createUser(User user) {
        // 这个方法调用了本类的其他方法
        validateUser(user); // 这里的事务会生效吗?
        saveUser(user); // 不会!
    }
    
    @Transactional
    public void saveUser(User user) {
        userDao.save(user);
    }
}
  1. 异常类型不正确:默认情况下,Spring只会在抛出运行时异常(RuntimeException)或Error时回滚事务。如果你抛出了检查异常(比如Exception),事务是不会回滚的。

二、Spring事务传播行为的原理

事务传播行为定义了一个事务方法被另一个事务方法调用时,事务应该如何传播。Spring提供了7种传播行为,但最常用的是以下三种:

  1. REQUIRED(默认):如果当前存在事务,就加入该事务;如果当前没有事务,就新建一个事务。这是最常用的传播行为。

这就好像朋友聚餐:如果有人已经订了包间(已有事务),你就直接加入;如果没人组织,你就自己订一个包间(新建事务)。

  1. REQUIRES_NEW:无论当前是否存在事务,都新建一个事务。如果当前有事务,就把当前事务挂起。

这好比公司开会时,老板突然要开个紧急小会:不管原来在开什么会,先暂停,开完小会再继续原来的会议。

  1. NESTED:如果当前存在事务,则在嵌套事务内执行;如果当前没有事务,则新建一个事务。

这有点像写文章时的"撤销"功能:你可以在一篇文章中做一些修改,如果不满意可以回滚到保存点,而不影响整篇文章。

三、工作中如何选择?

  • 大多数情况下,我们选择使用默认的REQUIRED就足够了
  • 当你需要确保某些操作无论如何都要提交,不受主事务回滚影响时,使用REQUIRES_NEW
  • 当你希望部分操作可以独立回滚而不影响主事务时,使用NESTED

四、小小建议

  1. 明确指定回滚异常:虽然Spring有默认行为,但最好明确指定回滚的异常类型,避免意外情况。
java 复制代码
@Transactional(rollbackFor = Exception.class)
  1. 只在需要的方法上使用事务:不要滥用@Transactional,因为事务是有性能开销的。

  2. 设置合适的事务超时时间:避免长时间占用数据库连接。

java 复制代码
@Transactional(timeout = 30)
  1. 读取操作设置为只读:如果你的方法只读取数据,设置readOnly = true可以提高性能。
java 复制代码
@Transactional(readOnly = true)

OK!本次的分享就到此结束,看看自己的项目中的@Transactional是否正确吧!

相关推荐
你曾经是少年1 分钟前
Java 关键字
java
海南java第二人4 分钟前
SpringBoot启动流程深度解析:从入口到容器就绪的完整机制
java·开发语言
问今域中6 分钟前
Spring Boot 请求参数绑定注解
java·spring boot·后端
星火开发设计8 分钟前
C++ queue 全面解析与实战指南
java·开发语言·数据结构·c++·学习·知识·队列
rgeshfgreh10 分钟前
Java+GeoTools+PostGIS高效求解对跖点
java
鱼跃鹰飞11 分钟前
DDD中的防腐层
java·设计模式·架构
计算机程序设计小李同学13 分钟前
婚纱摄影集成管理系统小程序
java·vue.js·spring boot·后端·微信小程序·小程序
栈与堆1 小时前
LeetCode 19 - 删除链表的倒数第N个节点
java·开发语言·数据结构·python·算法·leetcode·链表
一路向北·重庆分伦1 小时前
03-01:MQ常见问题梳理
java·开发语言
一 乐1 小时前
绿色农产品销售|基于springboot + vue绿色农产品销售系统(源码+数据库+文档)
java·前端·数据库·vue.js·spring boot·后端·宠物