Spring 事务和事务传播机制

简介: Spring 事务和事务传播机制

一、再谈事务

到这里 JavaEE 的学习基本是已经接近了尾声,相信大家对事务已然有了一些理解。当然这里我们还是简单的说明一下:

事务就是将一组操作封装成一个执行单元,要么全部成功,要么全部失败。比较典型的应用场景是转账,可想而知,跟¥挂钩的都是非常重要的,容不得一点闪失,转账要么成功要么失败,不能存在其他情况。

二、Spring 中事务实现

Spring 中事务的实现主要分为两类:

  1. 编程式事务(手动写代码操作事务)。
  2. 声明式事务(利用注解自动开启和提交事务)。

编程式事务主要分为3个步骤:开启事务、提交事务、回滚事务。操作比较繁琐,开发效率较低。而我们实际开发中常常使用声明式事务,即使用添加注解的方式实现上述过程。下面我们就围绕 声明式事务 展开讲解。

1、Spring 声明式事务概述

声明式事务的实现很简单,只需要在需要的方法上添加 @Transactional 注解就可以实现了,无需手动开启事务和提交事务,进入方法时自动开启事务,方法执行完会自动提交事务,如果中途发生了没有处理的异常 会自动回滚事务。

下面是使用 @Transactional 注解完成异常回滚的示例:

@RestController
@RequestMapping("/user")
public class UserController {
  // 属性注入
    @Autowired
    private UserService userService;

    @RequestMapping("/insert")
    @Transactional
    public int insert() {
        // 这里构造一个测试用例
        UserInfo userInfo = new UserInfo();
        userInfo.setUsername("张三");
        userInfo.setPassword("666");

        // 调用 service 接口
        int result = userService.insert(userInfo);

        // 添加异常
        int a = 10/0;
        // 返回结果
        return result;
    }
}

2、@Transactional 作用范围

@Transactional 可以用来修饰方法或类:

  • 修饰方法时:需要注意只能应用到 public 方法上,否则不生效。
  • 修饰类时:表明该注解对该类中所有的 public 方法都生效。

3、@Transactional 参数说明

下表是 @Transactional 中的所有参数:

参数 作用
value 当配置了多个事务管理器时,可以使用该属性指定选择哪个事务管理器
transactionManager 当配置了多个事务管理器时,可以使用该属性指定选择哪个事务管理器
propagation 事务的传播行为,默认值为 Propagation.REQUIRED
isolation 事务的隔离级别,默认值为 Isolation.DEFAULT
timeout 事务的超时时间,默认值为 -1。如果超过该时间限制但事务还没有完成,则自动回滚事务
readOnly 指定事务是否为只读事务,默认值为 false。为了忽略那些不需要事务的方法,比如读取数据可以设置 readOnly 为 true
rollbackFor 用于指定能够触发事务回滚的异常类型,可以指定多个异常类型
rollbackForClassName 用于指定能够触发事务回滚的异常类型,可以指定多个异常类型(通过类名指定)
noRollbackFor 抛出指定的异常类型,不回滚事务,也可以指定多个异常类型
noRollbackForClassName 抛出指定的异常类型,不回滚事务,也可以指定多个异常类型(通过类名指定)

其中有两个加粗显示的参数,分别是 propagation 表示事务的传播行为;isolation 表示事务的隔离级别。加粗自然就比较重要,下面我们分别对这两个参数展开讲解:

4、Spring 事务隔离级别

我们知道事务有 ACID 四大特性:原子性(Atomicity)、持久性(Durability)、一致性(Consistency) 和 隔离性(Isolation)。但是这四个特性中,只有 隔离性 是可以设置的。


设置事务的隔离级别是用来保障多个并发事务执行更可控,就是为了防止,其他的事务影响当前事务执行的一种策略。

对于我们熟悉的 MySQL 来说,它的事务隔离级别主要有种:

事务隔离级别 脏读 不可重复读 幻读
读未提交 (READ UNCOMMITTED)
读已提交 (READ COMMITTED) ×
可重复读 (REPEATABLE READ) × ×
串行化 (SERIALIZABLE) × × ×

而在 Spring 中,可设置的事务隔离级别有种:

  1. Isolation.DEFAULT:以连接的数据库的事务隔离级别为主。
  2. Isolation.READ_UNCOMMITTED:读未提交,可以读取到未提交的事务,存在脏读。
  3. Isolation.READ_COMMITTED:读已提交,只能读取到已经提交的事务,解决了脏读,存在不可重复读。
  4. Isolation.REPEATABLE_READ:可重复读,解决了不可重复读,但存在幻读(MySQL默认级别)。
  5. Isolation.SERIALIZABLE:串行化,可以解决所有并发问题,但性能太低。

上面我们了解了 isolation 属性,在 Spring 中设置事务隔离级别只需要设置 @Transactional 里的 isolation 属性即可:

@RequestMapping("/insert")
    @Transactional(isolation = Isolation.DEFAULT)
    public int insert() {
        //...
    }

5、Spring 事务传播机制

事务的传播机制就是规定多个事务在相互调用时,事务的执行行为。Spring 中支持以下七种事务传播机制:

  1. Propagation.REQUIRED:默认的事务传播级别,如果当前方法没有事务,新建一个事务,如果已经存在一个事务,则加入到这个事务中。
  2. Propagation.SUPPORTS:如果当前存在事务,则加⼊该事务,如果当前没有事务,就以非事务方式执行。
  3. Propagation.MANDATORY:如果当前存在事务,则加⼊该事务,如果当前没有事务,就抛出异常。
  4. Propagation.REQUIRES_NEW:新建事务执行,如果当前存在事务,就把当前事务挂起。也就是说不管外部方法是否开启事务,Propagation.REQUIRES_NEW 修饰的内部方法会新开启自己的事务,且开启的事务相互独立,互不干扰。如果外部事务发生异常并回滚,标记为 REQUIRES_NEW 的内部事务不会受到外部事务的影响而回滚。
  5. Propagation.NOT_SUPPORTED:以非事务方式执行操作,如果当前存在事务,就把当前事务挂起。
  6. Propagation.NEVER:以非事务方式执行,如果当前存在事务,则抛出异常。
  7. Propagation.NESTED:如果当前存在事务,则在嵌套事务内执行,如果当前没有事务,则执行等价于 Propagation.REQUIRED。如果外部事务发生异常并回滚,标记为 NESTED 的内部事务会共享外部事务的回滚

以上 7 种传播行为,可以根据是否支持当前事务分为以下 3 类:

在 Spring 中设置事务传播机制只需要设置 @Transactional 里的 propagation 属性即可。下面演示Propagation.NESTED 事务传播:

它们之间的嵌套关系如下:

访问 http://localhost:8080/user/insert 得到如下结果:

如果我们将上述异常代码删除,得到下面结果:

6、@Transactional 工作原理

@Transactional 是基于 AOP 实现的,AOP 用是使用态代理实现的。如果目标对象实现了接口,默认情况下会采用 JDK 的动态代理,如果目标对象没有实现了接口,会使用 CGLIB 动态代理。@Transactional 在开始执行业务之前,通过代理先开启事务,在执行成功之后再提交事务。如果中途遇到未处理的异常,则回滚事务。

@Transactional 具体执行细节如下:

注意:这里说的是“未处理”的异常,也就是没有使用 try-catch 进行异常处理。如果在出现异常的逻辑中,使用 try-catch 进行异常捕获,那么 AOP 层面就感知不到异常了,自然也就不会进行回滚操作。此时我们有两种解决方案:


方案一:在 try-catch 中重新将异常抛出

 try {
   // 执⾏了异常代码(0不能做除数)
   int i = 10 / 0;
 } catch (Exception e) {
   System.out.println(e.getMessage());
   // 将异常重新抛出去
   throw e;
 }

方案二:在 try-catch 中手动回滚事务

try {
   // 执⾏了异常代码(0不能做除数)
   int i = 10 / 0;
 } catch (Exception e) {
   System.out.println(e.getMessage());
   // ⼿动回滚事务
   TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();
 }
相关文章
|
13天前
|
Java Spring
Spring中事务失效的场景
因为Spring事务是基于代理来实现的,所以某个加了@Transactional的⽅法只有是被代理对象调⽤时, 那么这个注解才会⽣效 , 如果使用的是被代理对象调用, 那么@Transactional会失效 同时如果某个⽅法是private的,那么@Transactional也会失效,因为底层cglib是基于⽗⼦类来实现 的,⼦类是不能重载⽗类的private⽅法的,所以⽆法很好的利⽤代理,也会导致@Transactianal失效 如果在业务中对异常进行了捕获处理 , 出现异常后Spring框架无法感知到异常, @Transactional也会失效
|
12天前
|
Java 关系型数据库 数据库
微服务——SpringBoot使用归纳——Spring Boot事务配置管理——常见问题总结
本文总结了Spring Boot中使用事务的常见问题,虽然通过`@Transactional`注解可以轻松实现事务管理,但在实际项目中仍有许多潜在坑点。文章详细分析了三个典型问题:1) 异常未被捕获导致事务未回滚,需明确指定`rollbackFor`属性;2) 异常被try-catch“吃掉”,应避免在事务方法中直接处理异常;3) 事务范围与锁范围不一致引发并发问题,建议调整锁策略以覆盖事务范围。这些问题看似简单,但一旦发生,排查难度较大,因此开发时需格外留意。最后,文章提供了课程源代码下载地址,供读者实践参考。
31 0
|
12天前
|
Java 关系型数据库 数据库
微服务——SpringBoot使用归纳——Spring Boot事务配置管理——Spring Boot 事务配置
本文介绍了 Spring Boot 中的事务配置与使用方法。首先需要导入 MySQL 依赖,Spring Boot 会自动注入 `DataSourceTransactionManager`,无需额外配置即可通过 `@Transactional` 注解实现事务管理。接着通过创建一个用户插入功能的示例,展示了如何在 Service 层手动抛出异常以测试事务回滚机制。测试结果表明,数据库中未新增记录,证明事务已成功回滚。此过程简单高效,适合日常开发需求。
45 0
|
12天前
|
Java 数据库 微服务
微服务——SpringBoot使用归纳——Spring Boot事务配置管理——事务相关
本文介绍Spring Boot事务配置管理,阐述事务在企业应用开发中的重要性。事务确保数据操作可靠,任一异常均可回滚至初始状态,如转账、购票等场景需全流程执行成功才算完成。同时,事务管理在Spring Boot的service层广泛应用,但根据实际需求也可能存在无需事务的情况,例如独立数据插入操作。
19 0
|
13天前
|
SQL Java 数据库连接
Spring中的事务是如何实现的
1. Spring事务底层是基于数据库事务和AOP机制的 2. ⾸先对于使⽤了@Transactional注解的Bean,Spring会创建⼀个代理对象作为Bean 3. 当调⽤代理对象的⽅法时,会先判断该⽅法上是否加了@Transactional注解 4. 如果加了,那么则利⽤事务管理器创建⼀个数据库连接 5. 并且修改数据库连接的autocommit属性为false,禁⽌此连接的⾃动提交,这是实现Spring事务⾮ 常重要的⼀步 6. 然后执⾏当前⽅法,⽅法中会执⾏sql 7. 执⾏完当前⽅法后,如果没有出现异常就直接提交事务 8. 如果出现了异常,并且这个异常是需要回滚的就会回滚事务
|
16天前
|
JavaScript Java 开发者
Spring事务失效,常见的情况有哪些?
本文总结了Spring事务失效的7种常见情况,包括未启用事务管理功能、方法非public类型、数据源未配置事务管理器、自身调用问题、异常类型错误、异常被吞以及业务和事务代码不在同一线程中。同时提供了两种快速定位事务相关Bug的方法:通过查看日志(设置为debug模式)或调试代码(在TransactionInterceptor的invoke方法中设置断点)。文章帮助开发者更好地理解和解决Spring事务中的问题。
|
3月前
|
SQL Java 关系型数据库
【SpringFramework】Spring事务
本文简述Spring中数据库及事务相关衍伸知识点。
58 9
|
4月前
|
Java 开发者 Spring
理解和解决Spring框架中的事务自调用问题
事务自调用问题是由于 Spring AOP 代理机制引起的,当方法在同一个类内部自调用时,事务注解将失效。通过使用代理对象调用、将事务逻辑分离到不同类中或使用 AspectJ 模式,可以有效解决这一问题。理解和解决这一问题,对于保证 Spring 应用中的事务管理正确性至关重要。掌握这些技巧,可以提高开发效率和代码的健壮性。
228 13
|
4月前
|
缓存 安全 Java
Spring高手之路26——全方位掌握事务监听器
本文深入探讨了Spring事务监听器的设计与实现,包括通过TransactionSynchronization接口和@TransactionalEventListener注解实现事务监听器的方法,并通过实例详细展示了如何在事务生命周期的不同阶段执行自定义逻辑,提供了实际应用场景中的最佳实践。
120 2
Spring高手之路26——全方位掌握事务监听器
|
4月前
|
Java 关系型数据库 数据库
京东面试:聊聊Spring事务?Spring事务的10种失效场景?加入型传播和嵌套型传播有什么区别?
45岁老架构师尼恩分享了Spring事务的核心知识点,包括事务的两种管理方式(编程式和声明式)、@Transactional注解的五大属性(transactionManager、propagation、isolation、timeout、readOnly、rollbackFor)、事务的七种传播行为、事务隔离级别及其与数据库隔离级别的关系,以及Spring事务的10种失效场景。尼恩还强调了面试中如何给出高质量答案,推荐阅读《尼恩Java面试宝典PDF》以提升面试表现。更多技术资料可在公众号【技术自由圈】获取。