Spring事务传播机制

简介: 通过举例Spring 父方法、子方法调用以及事务传播机制验证

Spring事务传播类型

本文主要是讲述Spring事务传播机制,也就是大家开发过程中多多少少会遇到的父方法调用子方法,或者A方法调用B方法时事务的传播机制问题,但是在开始讲Spring事务传播机制之前需要先讲一下Spring事务的几种传播类型,如图

image.png

REQUIRED

REQUIRED表示如果当前没有事务,就创建一个事务,如果已经存在一个事务,就加入该事务,是Spring默认的事务传播类型

image.png

也就是说如果外部不存在事务,就开启新的事务,如果外部存在事务,就加入该事务中,如果调用者发生异常,那么调用者和被调用者的事务都回滚。

SUPPORTS

SUPPORTS表示支持当前事务,如果当前没有事务,就以非事务的方式执行,也就是说外部不存在事务,不会开启新事务,外部存在事务时就加入该事务。

MANDATORY

MANDATORY表示支持当前事务,但是跟SUPPORTS不同的是这种事务传播类型具备强制性,当前操作必须存在事务,如果不存在,则抛出异常。

REQUIRES_NEW

REQUIRES_NEW表示如果当前存在事务,则把当前事务挂起,重新创建新的事务并执行,知道新的事务提交或回滚,才会恢复执行原来的事务。这种事务传播类型新创建的事务和被挂起的事务没有任何关系,他们是两个相互独立的事务,外部事务失败后回滚,不会回滚内部事务的执行结果,内部事务执行失败抛出异常,被外部事务捕获时,外部事务可以不处理内部事务的回滚操作。

NOT_SUPPORTED

NOT_SUPPORTED表示以非事务方式执行,如果当前操作在一个事务中,就把当前事务挂起,以非事务方式运行,直到操作完成再恢复事务执行。

NEVER

NEVER表示以非事务的方式执行,如果当前操作存在事务,则抛出异常。

NESTED

NESTED表示如果当前方法有一个事务正在运行,则这个方法应该运行在一个嵌套事务中,被嵌套的事务可以独立于被封装的事务进行提交或回滚,也就是说如果封装事务存在,并且外层事务异常回滚,那么内层事务必须回滚,如果内层事务回滚,不影响外层事务的提交和回滚,当封装事务不存在时,按照REQUIRED事务传播类型执行。

总结

上面7种事务传播类型,日常用到的基本上就REQUIRED、REQUIRES_NEW、NOT_SUPPORTED这几种。

Spring事务举例

现在有这样一种场景,权限管理系统保存角色的时候同时保存角色绑定的菜单,下面开始测试,数据库角色表、角色绑定菜单表 初始数据情况 sys_role、sys_role_menu

image.png


无事务情况

父方法 insertRole保存角色,子方法 insertRoleMenu绑定角色菜单,代码如图

image.png

设置insertRoleMenu 内部发生异常,则insertRole保存角色(普通角色)成功,insertRoleMenu绑定角色菜单失败,验证结果

image.png

父方法未开启事务,子方法开启事务

父方法未开启事务,子方法开启默认事务传播类型REQUIRED,代码如图

image.png

设置insertRoleMenu 内部发生异常,同时insertRoleMenu 开启Spring默认事务,而insertRole未添加事务注解,则insertRole保存角色(普通角色1)成功,insertRoleMenu 绑定角色菜单内部发生异常,事务回滚,验证结果

image.png

父方法开启事务,子方法未开启事务

父方法开启默认事务传播类型,子方法未开启事务,代码如图

image.png

设置insertRoleMenu 内部发生异常,同时insertRole 开启Spring默认事务,而insertRoleMenu未添加事务注解,则insertRoleMenu 内部发生异常,会影响外部insertRole方法的执行,此时均发生回滚,验证结果

image.png

新增(普通角色2)异常回滚,绑定角色菜单异常回滚。

父方法开启事务,子方法开启事务

父方法开启Spring默认事务传播类型,子方法开启Spring默认事务传播类型

image.png

设置insertRoleMenu 内部发生异常,同时insertRole 开启Spring默认事务,insertRoleMenu开启Spring默认事务,则insertRoleMenu 内部发生异常,会影响外部insertRole方法的执行,此时均发生回滚,验证结果

image.png

新增(普通角色3)异常回滚,绑定角色菜单异常回滚。

父方法开启REQUIRED事务,子方法开启NOT_SUPPORTED事务

父方法开启Spring默认事务传播类型,子方法开启NOT_SUPPORTED事务

image.png

设置insertRoleMenu 内部发生异常,同时insertRole 开启Spring默认事务,insertRoleMenu开启Spring NOT_SUPPORTED事务,由于insertRoleMenu将父方法事务挂起,且执行绑定角色菜单之后发生异常,则insertRoleMenu会执行成功,同时由于子方法发生异常,则会影响外部insertRole方法的执行,则insertRole执行失败,事务回滚,验证结果

image.png

新增(普通角色4)增加失败,绑定角色菜单执行成功。

父方法开启REQUIRED事务,子方法开启REQUIRES_NEW事务

父方法开启Spring默认事务传播类型,子方法开启Spring REQUIRES_NEW事务传播类型

image.png

设置insertRoleMenu 内部发生异常,同时insertRole 开启Spring默认事务,insertRoleMenu开启Spring REQUIRES_NEW事务,此时insertRoleMenu发生异常时,则insertRoleMenu、insertRole均会执行失败,事务回滚,验证结果

image.png

新增(普通角色5)增加失败,绑定角色菜单执行失败,事务回滚。

父方法开启REQUIRED事务,子方法开启REQUIRES_NEW事务

父方法开启Spring默认事务传播类型,子方法开启Spring REQUIRES_NEW事务传播类型,与上一种不同的是,异常产生发生在父方法最后一行

image.png

设置insertRole 内部发生异常,且异常发生在最后一行,同时insertRole 开启Spring默认事务,insertRoleMenu开启Spring REQUIRES_NEW事务,此时insertRole发生异常时,则insertRole均会执行失败,事务回滚,insertRoleMenu执行成功,验证结果

image.png

新增(普通角色6)增加失败,绑定角色菜单执行成功。

父方法开启REQUIRED事务,子方法开启REQUIRES_NEW事务

父方法开启Spring默认事务传播类型,子方法开启Spring REQUIRES_NEW事务传播类型,异常产生发生在父方法最后一行,同时父方法、子方法是在同一个类中的内部调用

image.png

设置insertRole 内部发生异常,且异常发生在最后一行,同时insertRole 开启Spring默认事务,insertRoleMenu开启Spring REQUIRES_NEW事务,此时insertRole发生异常时,则insertRole、insertRoleMenu执行失败,事务回滚,验证结果

image.png

新增(普通角色7)增加失败,绑定角色菜单执行失败。

Spring 事务失效场景

数据库不支持事务

Spring事务生效的前提是连接的数据库存储引擎支持事务,比如Mysql 的MyISAM存储引擎不支持事务,则Spring事务会失效。

事务方法未被Spring管理

如果事务所在的类没有被交给Spring 管理,或者说没有加载到Spring IOC容器中,则该类方法上的事务会失效,比如

image.png

如果没有@Service注解,则该类下的方法上的事务会失效。

事务方法私有private

如果事务注解的方法为内部的私有private方法,则事务会失效。

同一个类中的方法调用

同一个类中的父方法调用子方法,都开启了事务,如图

image.png

子方法的事务会失效

不正确的异常捕获

如果发生异常的代码被try catch捕获,则Spring 管理器无法感知到异常,导致事务失效。

错误的异常类型

Spring 事务默认的捕获的异常是RuntimeException,如果代码中执行失败抛出了Exception异常,则事务会失效


总结

以上内容基于日常工作及学习,仅供参考,有兴趣的同学也可以买一本《深入理解分布式事务 原理与实战》,读之受益良多。


相关文章
|
3月前
|
安全 Java 数据库
一天十道Java面试题----第四天(线程池复用的原理------>spring事务的实现方式原理以及隔离级别)
这篇文章是关于Java面试题的笔记,涵盖了线程池复用原理、Spring框架基础、AOP和IOC概念、Bean生命周期和作用域、单例Bean的线程安全性、Spring中使用的设计模式、以及Spring事务的实现方式和隔离级别等知识点。
|
5天前
|
Java 开发者 Spring
Spring高手之路24——事务类型及传播行为实战指南
本篇文章深入探讨了Spring中的事务管理,特别是事务传播行为(如REQUIRES_NEW和NESTED)的应用与区别。通过详实的示例和优化的时序图,全面解析如何在实际项目中使用这些高级事务控制技巧,以提升开发者的Spring事务管理能力。
15 1
Spring高手之路24——事务类型及传播行为实战指南
|
4月前
|
Java 关系型数据库 MySQL
Spring 事务失效场景总结
Spring 事务失效场景总结
61 4
|
2月前
|
Java 数据库连接 数据库
spring复习05,spring整合mybatis,声明式事务
这篇文章详细介绍了如何在Spring框架中整合MyBatis以及如何配置声明式事务。主要内容包括:在Maven项目中添加依赖、创建实体类和Mapper接口、配置MyBatis核心配置文件和映射文件、配置数据源、创建sqlSessionFactory和sqlSessionTemplate、实现Mapper接口、配置声明式事务以及测试使用。此外,还解释了声明式事务的传播行为、隔离级别、只读提示和事务超时期间等概念。
spring复习05,spring整合mybatis,声明式事务
|
2月前
|
Java 测试技术 数据库
Spring事务传播机制(最全示例)
在使用Spring框架进行开发时,`service`层的方法通常带有事务。本文详细探讨了Spring事务在多个方法间的传播机制,主要包括7种传播类型:`REQUIRED`、`SUPPORTS`、`MANDATORY`、`REQUIRES_NEW`、`NOT_SUPPORTED`、`NEVER` 和 `NESTED`。通过示例代码和数据库插入测试,逐一展示了每种类型的运作方式。例如,`REQUIRED`表示如果当前存在事务则加入该事务,否则创建新事务;`SUPPORTS`表示如果当前存在事务则加入,否则以非事务方式执行;`MANDATORY`表示必须在现有事务中运行,否则抛出异常;
121 4
Spring事务传播机制(最全示例)
|
1月前
|
Java 关系型数据库 MySQL
Spring事务失效,我总结了这7个主要原因
本文详细探讨了Spring事务在日常开发中常见的七个失效原因,包括数据库不支持事务、类不受Spring管理、事务方法非public、异常被捕获、`rollbackFor`属性配置错误、方法内部调用事务方法及事务传播属性使用不当。通过具体示例和源码分析,帮助开发者更好地理解和应用Spring事务机制,避免线上事故。适合所有使用Spring进行业务开发的工程师参考。
28 2
|
1月前
|
Java 程序员 Spring
Spring事务的1道面试题
每次聊起Spring事务,好像很熟悉,又好像很陌生。本篇通过一道面试题和一些实践,来拆解几个Spring事务的常见坑点。
Spring事务的1道面试题
|
2月前
|
Java Spring
Spring 事务传播机制是什么?
Spring 事务传播机制是什么?
22 4
|
1月前
|
监控 Java 数据库
Spring事务中的@Transactional注解剖析
通过上述分析,可以看到 `@Transactional`注解在Spring框架中扮演着关键角色,它简化了事务管理的复杂度,让开发者能够更加专注于业务逻辑本身。合理运用并理解其背后的机制,对于构建稳定、高效的Java企业应用至关重要。
43 0
|
3月前
|
Java 程序员 数据库连接
女朋友不懂Spring事务原理,今天给她讲清楚了!
该文章讲述了如何解释Spring事务管理的基本原理,特别是针对女朋友在面试中遇到的问题。文章首先通过一个简单的例子引入了传统事务处理的方式,然后详细讨论了Spring事务管理的实现机制。
女朋友不懂Spring事务原理,今天给她讲清楚了!
下一篇
无影云桌面