从@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是否正确吧!

相关推荐
fatfishccc1 小时前
Spring MVC 全解析:从核心原理到 SSM 整合实战 (附完整源码)
java·spring·ajax·mvc·ssm·过滤器·拦截器interceptor
没有bug.的程序员1 小时前
MyBatis 初识:框架定位与核心原理——SQL 自由掌控的艺术
java·数据库·sql·mybatis
执键行天涯1 小时前
从双重检查锁定的设计意图、锁的作用、第一次检查提升性能的原理三个角度,详细拆解单例模式的逻辑
java·前端·github
程序员江鸟1 小时前
Java面试实战系列【JVM篇】- JVM内存结构与运行时数据区详解(私有区域)
java·jvm·面试
回家路上绕了弯2 小时前
ClickHouse 深度解析:从核心特性到实战应用,解锁 OLAP 领域新势能
数据库·后端
架构师沉默2 小时前
Java 状态机设计:替代 if-else 的优雅架构
java·程序员·架构
java亮小白19972 小时前
Spring Cloud 快速通关之Sentinel
java·spring cloud·sentinel
atwednesday2 小时前
大规模文档预览的架构设计与实现策略
java
xiaok2 小时前
本地用VScode的Live Server监听5500访问页面,ubuntu上不需要在配置5500
后端
雨绸缪2 小时前
ABAP 时间戳
后端