Spring 事务的使用、隔离级别、@Transactional的使用

Spring事务是Spring框架提供的一种机制,用于管理应用程序中的数据库事务。
事务是一组数据库操作的执行单元,要么全部成功提交,要么全部失败回滚,保证数据的一致性和完整性。
Spring事务提供了声明式事务和编程式事务两种方式:
编程式事务: 编程式事务是通过编写代码来手动管理事务的开始、提交或回滚,开发人员需要显式地调用事务管理器的方法来控制事务的边界。
声明式事务: 声明式事务是通过在配置文件或注解中声明事务的属性来实现的。开发人员只需要关注业务逻辑,而不需要关心事务的管理。Spring框架会根据注解的配置自动管理事务的开始、提交或回滚。
1. 编程式事务:
在Spring编程式事务中,DataSourceTransactionManager和TransactionDefinition是两个核心的类:
DataSourceTransactionManager: 它是基于数据源(DataSource)的事务管理器,通过获取连接并控制连接的提交或回滚来管理事务。使用时,需要配置一个数据源(DataSource),用于获取数据库连接。在事务开始时,它会从数据源中获取一个数据库连接,并将该连接绑定到当前线程上下文中。在事务结束时,它会根据事务的提交或回滚状态,来决定是否提交或回滚数据库连接。
TransactionDefinition: 是Spring框架定义的事务属性接口,用于定义事务的一些属性,比如事务的隔离级别、传播行为、超时时间等。
@RestController
@RequestMapping("/user")
public class UserController {@Autowiredprivate UserService userService;@Autowiredprivate DataSourceTransactionManager transactionManager;@Autowiredprivate TransactionDefinition transactionDefinition;@RequestMapping("/del")public int del(Integer id) {if (id != null && id > 0) {// 开启事务TransactionStatus transactionStatus = null;// 业务操作,进行删除int result = 0;try {transactionStatus = transactionManager.getTransaction(transactionDefinition);result = userService.del(id);System.out.println("删除成功:" + result);transactionManager.commit(transactionStatus); // 提交} catch (Exception e) {if (transactionStatus != null) {transactionManager.rollback(transactionStatus); // 回滚}}return result;}return 0;}
}
2. 声明式事务:
编程式事务写起来较麻烦,声明式事务用起来较方便。
声明式事务可以通过使用 @Transactional 注解来实现:
当方法执行完成后,如果没有发生异常,事务管理器会自动提交事务;如果发生异常,事务管理器会自动回滚事务。也可以通过在方法上抛出异常手动来触发事务的回滚。
@RestController
@RequestMapping("/user2")
public class UserController2 {@Autowiredprivate UserService userService;@Transactional@RequestMapping("del")public int del(Integer id) {if (id == null || id < 0) return 0;int ret = userService.del(id);System.out.println("删除:" + id);return ret;}
}
3. @Transactional 注解的具体使用:
3.1 作用范围:
@Transactional注解支持修饰方法,也可以修饰类:
修饰方法时:需要注意只能应用到 public 方法上,否则不生效。推荐此种用法。
修饰类时:表明该注解对该类中所有的 public 方法都生效。
3.2 支持参数设置:
这些参数可以根据实际需求进行设置,如果不指定参数,则使用默认值。
value:指定使用的事务管理器的名称。可以通过名称引用已经配置的事务管理器。默认值为空,表示使用默认的事务管理器。
propagation:指定事务的传播行为。可以设置为以下值之一:
- REQUIRED:如果当前没有事务,则创建一个新的事务;如果当前存在事务,则加入到当前事务中。
- REQUIRES_NEW:创建一个新的事务,如果当前存在事务,则将当前事务挂起。
- NESTED:在当前事务的嵌套事务中执行。如果当前没有事务,则创建一个新的事务。
- SUPPORTS:如果当前存在事务,则加入到当前事务中;如果当前没有事务,则以非事务方式执行。
- MANDATORY:如果当前存在事务,则加入到当前事务中;如果当前没有事务,则抛出异常。
- NEVER:以非事务方式执行,如果当前存在事务,则抛出异常。
- NOT_SUPPORTED:以非事务方式执行,如果当前存在事务,则将当前事务挂起。
isolation:指定事务的隔离级别。可以设置为以下值之一:
- DEFAULT:使用默认的隔离级别(以连接的数据库隔离级别为准)。
- READ_UNCOMMITTED:允许读取未提交的数据。
- READ_COMMITTED:只能读取已提交的数据。
- REPEATABLE_READ:保证可重复读取,禁止脏读和不可重复读。
- SERIALIZABLE:保证可串行化的隔离级别,禁止脏读、不可重复读和幻读。
timeout:指定事务的超时时间,单位为秒。如果事务在指定的时间内没有完成,则会被自动回滚。默认值为-1,表示没有超时限制。
readOnly:指定事务是否为只读事务。如果设置为true,则表示只读事务,不允许进行任何修改操作。默认值为false。
rollbackFor:指定哪些异常触发事务的回滚。可以设置为一个异常类或异常类数组。默认情况下,只有运行时异常和Error会触发事务的回滚。
noRollbackFor:指定哪些异常不触发事务的回滚。可以设置为一个异常类或异常类数组。默认情况下,运行时异常和Error会触发事务的回滚。
如图: 如果要加多个参数,可以使用逗号隔开

使用注意事项:未检测到异常不回滚
使用 @Transactional 注解是它检测到异常才会回滚,但如果在代码中,使用 try-catch 检测到异常,并在这时程序终止了,那么事务就不会被回滚,这时非常可怕的事情。
例如:

解决方案1:从原因入手
既然是 @Transactional 没有检测到异常,那我们让它检测而到就可以了,所以可以在 catch 中将异常抛出,让 @Transactional 能检测到即可

解决方案2:从结果入手
在 catch 中借助 TransactionAspectSupport 的相关方法来手动回滚事务

3.3 经典面试题:
(1)@Transactional 的工作原理?
@Transactional是Spring框架中用于实现声明式事务管理的注解。它的工作原理是基于AOP(Aspect-Oriented Programming)技术。
当Spring容器启动时,会扫描所有被@Transactional注解标注的类和方法。
当调用被@Transactional注解标注的方法时,Spring会通过AOP技术在方法执行前后织入事务管理的逻辑。
在方法执行前,事务管理器会开启一个新的事务。
在方法执行过程中,如果发生异常,事务管理器会回滚事务,即撤销之前的操作。
如果方法执行成功,事务管理器会提交事务,即将之前的操作永久保存到数据库中。
如果方法执行过程中调用了其他被@Transactional注解标注的方法,事务管理器会根据事务的传播行为来决定是否将这些方法加入到当前事务中。
需要注意的是,声明式事务只适用于公共方法,即在Spring容器中被代理的方法。如果在同一个类中的非公共方法中调用了带有@Transactional注解的公共方法,事务将不会生效。这是因为Spring使用AOP技术来实现声明式事务,只能拦截公共方法的调用。
(2)Spring 事务失效有哪些原因?
Spring事务失效的原因有以下几种可能:
1. 未正确配置事务管理器:在Spring中,需要配置事务管理器(例如DataSourceTransactionManager)来管理事务。如果未正确配置事务管理器,或者配置有误,可能导致事务失效。
2. 事务注解未生效:使用@Transactional注解来声明事务,但是注解未被正确识别和解析。可能是因为未在配置文件中启用事务注解的解析,或者注解所在的类未被Spring容器扫描到。
3. 异常未被正确抛出:事务的回滚是基于异常的,默认情况下,只有运行时异常和Error会触发事务的回滚。如果在事务方法中捕获了异常并未重新抛出,或者捕获的异常类型不符合回滚条件,事务将不会回滚。
4. 事务传播行为设置不正确:事务的传播行为决定了事务方法与其他事务方法的关系。如果事务方法的传播行为设置不正确,可能导致事务失效或不符合预期。
5. 方法未被代理:声明式事务是通过AOP技术实现的,只有被Spring容器管理的Bean中的方法才会被代理,从而实现事务的管理。如果方法未被代理,事务将不会生效。
6. 数据库不支持事务:某些数据库可能不支持事务,或者事务的隔离级别不被支持。在这种情况下,无法实现事务的管理。
7. 方法被重载:如果一个类中存在多个同名方法,但参数列表不同,而只有其中一个方法被@Transactional注解标注,那么其他同名方法的事务将不会生效。
相关文章:
Spring 事务的使用、隔离级别、@Transactional的使用
Spring事务是Spring框架提供的一种机制,用于管理应用程序中的数据库事务。 事务是一组数据库操作的执行单元,要么全部成功提交,要么全部失败回滚,保证数据的一致性和完整性。 Spring事务提供了声明式事务和编程式事务两种方式&am…...
Top命令
Top top - 12:46:01 up 2 days, 11:10, 3 users, load average: 0.56, 0.59, 0.45系统基本信息:显示了系统运行时间、登录用户数和平均负载(load average)情况。平均负载是系统在特定时间范围内的平均活跃进程数,可以用来衡量系…...
(三)RabbitMQ七种模式介绍与代码演示
Lison <dreamlison163.com>, v1.0.0, 2023.06.22 七种模式介绍与代码演示 文章目录 七种模式介绍与代码演示四大交换机四种交换机介绍 工作模式简单模式(Hello World)工作队列模式(Work queues)订阅模式(Publis…...
ElasticSearch Java API 操作
1.idea创建Maven项目 2.添加依赖 修改 pom.xml 文件 <dependency><groupId>org.elasticsearch</groupId><artifactId>elasticsearch</artifactId><version>7.8.0</version></dependency><!-- elasticsearch 的客户端 --…...
【Qt】QML-01:使用QtCreator10创建QML工程,并讲解第一个程序:Hello World
1、创建QML工程 1)新建工程 打开QtCreator10,依次点击“Create Project” --> “Application(Qt)” --> “Qt Quick Application(compat)” 注意:本人打算使用Qt5.15.2创建工程,而非Qt6,因此选择兼容低于Qt6版本的“Qt Quick Applicat…...
Docker的安装与部署
Docker 基本概念介绍 通俗理解:镜像是类,容器是对象实例 仓库 应用商店、镜像 下载的应用安装程序、容器 应用程序 镜像(Image) 这里面保存了应用和需要的依赖环境 为什么需要多个镜像?当开发、构建和运行容器化应用程序时,我们…...
【数据结构】实验四:循环链表
实验四 循环链表 一、实验目的与要求 1)熟悉循环链表的类型定义和基本操作; 2)灵活应用循环链表解决具体应用问题。 二、实验内容 题目一:有n个小孩围成一圈,给他们从1开始依次编号,从编号为1的小孩开…...
【FPGA/D7】
2023年7月26日 串口传图到RAM并TFT显示 视频25note要求:接收两个字节数据合并为一个16位数据并写入ram: FIFO模型与应用场景 视频26 串口传图到RAM并TFT显示 视频25 note 存储器的使用,在开始读写或者结束读写的位置非常容易出现数据错误或…...
Vue的下载以及MVVM分析
😀前言本片文章是vue系列第一篇整理了vue的基础和发展史 🏠个人主页:尘觉主页 🧑个人简介:大家好,我是尘觉,希望我的文章可以帮助到大家,您的满意是我的动力😉Ƕ…...
ElasticSearch学习--自动补全
目录 自定义分词器 介绍 配置自定义分词器 拼音分词器的问题编辑 总结 DSL自动补全查询 RestAPI实现自动补全 自定义分词器 介绍 自定义分词器只在当前库中有效 配置自定义分词器 拼音分词器的问题 总结 DSL自动补全查询 RestAPI实现自动补全...
【C++】多态,虚函数表相关问题解决
文章目录 多态概念及其触发条件重写和协变(考点1)(考点2) 虚函数表及其位置(考点3) 多继承中的虚函数表 多态概念及其触发条件 多态的概念:通俗来说,就是多种形态。具体点就是去完成…...
探索大型语言模型的开源人工智能基础设施:北京开源AI Meetup回顾
原文参见Explore open source AI Infra for Large Language Models: Highlights from the Open Source AI Meetup Beijing | Cloud Native Computing Foundation 背景介绍: 最近,在 ChatGPT 的成功推动下,大型语言模型及其应用程序的流行度激…...
Langchain 的 Conversation buffer window memory
Langchain 的 Conversation buffer window memory ConversationBufferWindowMemory 保存一段时间内对话交互的列表。它仅使用最后 K 个交互。这对于保持最近交互的滑动窗口非常有用,因此缓冲区不会变得太大。 我们首先来探讨一下这种存储器的基本功能。 示例代码&…...
电流源电路
3.3.3电流源电路 镜像电流源 电路 分析 仿真 比例电流源 电路 分析 仿真 加射极输出器的电流源1 电路 分析 仿真 加射极输出器的电流源2 电路 分析 仿真 威尔逊电流源 电路 分析 仿真...
iOS开发-CMMotionManager传感器陀螺仪
iOS开发-CMMotionManager传感器陀螺仪 之前开发中遇到需要使用陀螺仪判断是否拍照时候水平判断,如果没有水平拍照,则给出提示。方便用户拍照合适的题目图片。 一、CMMotionManager CMMotionManager是什么 CMMotionManager 是 Core Motion 库的核心类&…...
影刀下载,插件安装
1、下载 在影刀官网下载:www.yingdao.com 2、谷歌插件安装 参考: 影刀插件安装各种方式 浏览器安装插件说明 - 影刀帮助中心 安装说明:驱动外置 Chrome 需要安装插件,并且保证此插件处于开启状态 方式一:用户头…...
Linux的tcpdump命令详解
tcpdump 一款sniffer工具,是Linux上的抓包工具,嗅探器 补充说明 tcpdump命令 是一款抓包,嗅探器工具,它可以打印所有经过网络接口的数据包的头信息,也可以使用-w选项将数据包保存到文件中,方便以后分析。…...
springboot运行报错Failed to load ApplicationContext for xxx
Failed to load ApplicationContext for报错解决方法 报错Failed to load ApplicationContext for 报错Failed to load ApplicationContext for 网上找了一堆方法都尝试了还是没用 包括添加mapperScan,添加配置类 配置pom文件 [外链图片转存失败,源站可能有防盗链机…...
[SQL挖掘机] - 内连接: inner join
介绍: 内连接是一种多表连接方式,用于将两个或多个表中的数据通过共同的列值进行匹配,并返回满足连接条件的匹配行。简单来说,内连接能够将相关联的数据组合在一起,以便进行更复杂和全面的数据分析。 内连接的工作原理如下&…...
mysql(四)数据备份
目录 前言 一、概述 二、备份的类型 (一)物理与逻辑角度 (二)数据库备份策略角度 三、常见的备份方法 四、完整备份 (一)打包数据库文件备份 (二)备份工具备份 五、增量备份 六、操…...
OpenLayers 可视化之热力图
注:当前使用的是 ol 5.3.0 版本,天地图使用的key请到天地图官网申请,并替换为自己的key 热力图(Heatmap)又叫热点图,是一种通过特殊高亮显示事物密度分布、变化趋势的数据可视化技术。采用颜色的深浅来显示…...
如何在看板中体现优先级变化
在看板中有效体现优先级变化的关键措施包括:采用颜色或标签标识优先级、设置任务排序规则、使用独立的优先级列或泳道、结合自动化规则同步优先级变化、建立定期的优先级审查流程。其中,设置任务排序规则尤其重要,因为它让看板视觉上直观地体…...
渗透实战PortSwigger靶场-XSS Lab 14:大多数标签和属性被阻止
<script>标签被拦截 我们需要把全部可用的 tag 和 event 进行暴力破解 XSS cheat sheet: https://portswigger.net/web-security/cross-site-scripting/cheat-sheet 通过爆破发现body可以用 再把全部 events 放进去爆破 这些 event 全部可用 <body onres…...
376. Wiggle Subsequence
376. Wiggle Subsequence 代码 class Solution { public:int wiggleMaxLength(vector<int>& nums) {int n nums.size();int res 1;int prediff 0;int curdiff 0;for(int i 0;i < n-1;i){curdiff nums[i1] - nums[i];if( (prediff > 0 && curdif…...
江苏艾立泰跨国资源接力:废料变黄金的绿色供应链革命
在华东塑料包装行业面临限塑令深度调整的背景下,江苏艾立泰以一场跨国资源接力的创新实践,重新定义了绿色供应链的边界。 跨国回收网络:废料变黄金的全球棋局 艾立泰在欧洲、东南亚建立再生塑料回收点,将海外废弃包装箱通过标准…...
Python如何给视频添加音频和字幕
在Python中,给视频添加音频和字幕可以使用电影文件处理库MoviePy和字幕处理库Subtitles。下面将详细介绍如何使用这些库来实现视频的音频和字幕添加,包括必要的代码示例和详细解释。 环境准备 在开始之前,需要安装以下Python库:…...
JDK 17 新特性
#JDK 17 新特性 /**************** 文本块 *****************/ python/scala中早就支持,不稀奇 String json “”" { “name”: “Java”, “version”: 17 } “”"; /**************** Switch 语句 -> 表达式 *****************/ 挺好的ÿ…...
Linux 中如何提取压缩文件 ?
Linux 是一种流行的开源操作系统,它提供了许多工具来管理、压缩和解压缩文件。压缩文件有助于节省存储空间,使数据传输更快。本指南将向您展示如何在 Linux 中提取不同类型的压缩文件。 1. Unpacking ZIP Files ZIP 文件是非常常见的,要在 …...
作为测试我们应该关注redis哪些方面
1、功能测试 数据结构操作:验证字符串、列表、哈希、集合和有序的基本操作是否正确 持久化:测试aof和aof持久化机制,确保数据在开启后正确恢复。 事务:检查事务的原子性和回滚机制。 发布订阅:确保消息正确传递。 2、性…...
Rust 开发环境搭建
环境搭建 1、开发工具RustRover 或者vs code 2、Cygwin64 安装 https://cygwin.com/install.html 在工具终端执行: rustup toolchain install stable-x86_64-pc-windows-gnu rustup default stable-x86_64-pc-windows-gnu 2、Hello World fn main() { println…...
