Spring 事务和事务传播机制

一、再谈事务

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

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

二、Spring 中事务实现

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

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

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

1、Spring 声明式事务概述

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

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

@RestController
@RequestMapping("/user")
public class UserController {// 属性注入@Autowiredprivate UserService userService;@RequestMapping("/insert")@Transactionalpublic 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();}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/147368.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Vite - 配置 - 文件路径别名的配置

为什么要配置别名 别名的配置,主要作用是为了缩短代码中的导入路径。例如有如下的项目目录: project-name| -- src| -- a| --b| --c| --d| --e| -- abc.png| -- index.html| -- main.js如果想在 main.js 文件中使用 abc.png ,则使用的路径是 &#xff1…

二元分类模型评估方法

文章目录 前言一、混淆矩阵二、准确率三、精确率&召回率四、F1分数五、ROC 曲线六、AUC(曲线下面积)七、P-R曲线类别不平衡问题中如何选择PR与ROC 八、 Python 实现代码混淆矩阵、命中率、覆盖率、F1值ROC曲线、AUC面积 指标 公式 意义 真正例 (TP)被…

DNS1(Bind软件)

名词解释 1、DNS(Domain Name System) DNS即域名系统,它是一个分层的分布式数据库,存储着IP地址与主机名的映射 2、域和域名 域为一个标签,而有多个标签域构成的称为域名。例如hostname.example.com,其…

gRPC 的原理 介绍带你从头了解gRPC

gRPC 的原理 什么是gRPC gRPC的官方介绍是:gRPC是一个现代的、高性能、开源的和语言无关的通用 RPC 框架,基于 HTTP2 协议设计,序列化使用PB(Protocol Buffer),PB 是一种语言无关的高性能序列化框架,基于 HTTP2PB 保…

Sentinel 流控规则

Sentinel 是面向分布式、多语言异构化服务架构的流量治理组件,主要以流量为切入点,从流量路由、流量控制、流量整形、熔断降级、系统自适应过载保护、热点流量防护等多个维度来帮助开发者保障微服务的稳定性。 SpringbootDubboNacos 集成 Sentinel&…

实用技巧:在C和cURL中设置代理服务器爬取www.ifeng.com视频

概述: 网络爬虫技术作为一种自动获取互联网数据的方法,在搜索引擎、数据分析、网站监测等领域发挥着重要作用。然而,面对反爬虫机制、网络阻塞、IP封禁等挑战,设置代理服务器成为解决方案之一。代理服务器能够隐藏爬虫的真实IP地…

Word中NoteExpress不显示的问题

首先确认我们以及安装了word插件 我们打开word却没有。此时我们打开:文件->选项->加载项 我们发现被禁用了 选择【禁用项目】(如果没有,试一试【缓慢且禁用的加载项】),点击转到 选择启用 如果没有禁用且没有出…

C++加持让python程序插上翅膀——利用pybind11进行c++和python联合编程示例

目录 0、前言1、安装 pybind11库c侧python侧 2、C引入bybind11vs增加相关依赖及设置cpp中添加头文件及导出模块cpp中添加numpy相关数据结构的接收和返回编译生成dll后改成导出模块同名文件的.pyd 3、python调用c4、C引入bybind11 0、前言 在当今的计算机视觉和机器学习领域&am…

CSDN每日一题学习训练——Python版(简化路径,不同的二叉搜索树)

版本说明 当前版本号[20231116]。 版本修改说明20231116初版 目录 文章目录 版本说明目录简化路径题目解题思路代码思路参考代码 不同的二叉搜索树题目解题思路代码思路参考代码 简化路径 题目 给你一个字符串 path ,表示指向某一文件或目录的 Unix 风格 绝对路…

Mysql数据库 16.SQL语言 数据库事务

一、数据库事务 数据库事务介绍——要么全部成功要么全部失败 我们把完成特定的业务的多个数据库DML操作步骤称之为一个事务 事务——就是完成同一个业务的多个DML操作 例: 数据库事务四大特性 原子性(A):一个事务中的多个D…

(三)什么是Vite——Vite 主体流程(运行npm run dev后发生了什么?)

vite分享ppt,感兴趣的可以下载: ​​​​​​​Vite分享、原理介绍ppt 什么是vite系列目录: (一)什么是Vite——vite介绍与使用-CSDN博客 (二)什么是Vite——Vite 和 Webpack 区别&#xff0…

vscode 配置 lua

https://luabinaries.sourceforge.net/ 官网链接 主要分为4个步骤 下载压缩包,然后解压配置系统环境变量配置vscode的插件测试 这里你可以选择用户变量或者系统环境变量都行。 不推荐空格的原因是 再配置插件的时候含空格的路径 会出错,原因是空格会断…

linux 网络 cat /proc/net/dev 查看测试网络丢包情况

可以通过 cat /proc/net/dev 查看测试网络丢包情况,drop关键字,查看所有网卡的丢包情况 还可以看其他数据, /proc/net/下面有如下文件

性能测试 —— Jmeter接口处理不低于200次/秒-场景

需求:期望某个接口系统的处理能力不低于200次/秒,如何设计? ①这个场景是看服务器对某个接口的TPS值是否能大于等于200,就可以了; ②系统处理能力:说的就是我们性能测试中的TPS; ③只要设计一…

Visual Studio Code---介绍

0 Preface/Foreword 1、安装VScode 官网:Download Visual Studio Code - Mac, Linux, Windows 文档:Documentation for Visual Studio Code 1.1 优点 Intelligent code completion: code smarter with intellisense - completions for variables, me…

哈希

欢迎来到Cefler的博客😁 🕌博客主页:那个传说中的man的主页 🏠个人专栏:题目解析 🌎推荐文章:题目大解析(3) 目录 👉🏻unordered系列关联式容器un…

锐捷练习-ospf虚链路及rip路由相互引入

一、相关知识补充 1、ospf基本概述 OSPF(Open Shortest Path First)是一种链路状态路由协议,用于在计算机网络中进行路由选择。它是内部网关协议(IGP)之一,常用于大规模企业网络或互联网服务提供商的网络…

Pytorch torch.dot、torch.mv、torch.mm、torch.norm的用法详解

torch.dot的用法: 使用numpy求点积,对于二维的且一个二维的维数为1 torch.mv的用法: torch.mm的用法 torch.norm 名词解释:L2范数也就是向量的模,L1范数就是各个元素的绝对值之和例如:

flutter仿支付宝余额宝年化收益折线图

绘制: 1.在pubspec.yaml中引入:fl_chart: 0.55.2 2.绘制: import package:jade/utils/JadeColors.dart; import package:util/easy_loading_util.dart; import package:fl_chart/fl_chart.dart; import package:flutter/material.dart; impo…

微服务实战系列之Sentinel

前言 微服务架构(Microservice Architecture)是一种架构概念,旨在通过将功能分解到各个离散的服务中以实现对解决方案的解耦。 近年来,微服务已赫然崛起于IT界,越来越多的程序员不得不向之靠拢。也正因为各行各业都愿为…