Spring-事务管理-加强

news2025/1/20 14:51:08

目录

开启事务

编程式事务

声明式事务

声明式事务的优点

声明式事务的粒度问题

声明式事务用不对容易失效

Spring事务失效可能是哪些原因

@Transactional(rollbackFor = Exception.class)注解

Spring 事务的实现原理

事务传播机制

介绍

用法

rollbackFor

场景举例

事务接口


  • 开启事务

  • 事务管理在系统开发中是不可缺少的一部分,Spring提供了很好的事务管理机制,主要分为编程式事务和声明式事务两种
  • 编程式事务
  • 基于底层的API,如PlatformTransactionManager、TransactionDefinition 和 TransactionTemplate 等核心接口,开发者完全可以通过编程的方式来进行事务管理
  • 编程式事务方式需要开发者在代码中手动的管理事务的开启、提交、回滚等操作

  • 如以上代码,开发者可以通过API自己控制事务
  • 声明式事务
  • 声明式事务管理方法允许在开发者配置的帮助下来管理事务,而不需要依赖底层API进行硬编码
  • 开发者可以只使用注解或基于配置的 XML 来管理事务

  • 如上,使用@Transactional 即可给test方法增加事务控制
  • 声明式事务的优点
  • 通过上面的例子,其实我们可以很容易的看出来,声明式事务帮助我们节省了很多代码,他会自动帮我们进行事务的开启、提交以及回滚等操作,把程序员从事务管理中解放出来
  • 声明式事务管理是使用了 AOP 实现的,本质就是在目标方法执行前后进行拦截
  • 在目标方法执行前加入或创建一个事务,在执行方法执行后,根据实际情况选择提交或是回滚事务
  • 使用这种方式,对代码没有侵入性,方法内只需要写业务逻辑就可以了
  • 但是,声明式事务真的有这么好么?倒也不见得
  • 声明式事务的粒度问题
  • 首先,声明式事务有一个局限,那就是他的最小粒度要作用在方法上
  • 也就是说,如果想要给一部分代码块增加事务的话,那就需要把这个部分代码块单独独立出来作为一个方法
  • 但是,正是因为这个粒度问题,本人并不建议过度的使用声明式事务
  • 首先,因为声明式事务是通过注解的,有些时候还可以通过配置实现,这就会导致一个问题,那就是这个事务有可能被开发者忽略
  • 事务被忽略了有什么问题呢?
  • 首先,如果开发者没有注意到一个方法是被事务嵌套的,那么就可能会在方法中加入一些如RPC远程调用、消息发送、缓存更新、文件写入等操作
  • 我们知道,这些操作如果被包在事务中,有两个问题:
  • 1、这些操作自身是无法回滚的,这就会导致数据的不一致;可能RPC调用成功了,但是本地事务回滚了,可是RPC调用无法回滚了
  • 2、在事务中有远程调用,就会拉长整个事务;那么就会导致本事务的数据库连接一直被占用,那么如果类似操作过多,就会导致数据库连接池耗尽
  • 有些时候,即使没有在事务中进行远程操作,但是有些人还是可能会不经意的进行一些内存操作,如运算
  • 或者如果遇到分库分表的情况,有可能不经意间进行跨库操作
  • 但是如果是编程式事务的话,业务代码中就会清清楚楚看到什么地方开启事务,什么地方提交,什么时候回滚
  • 这样有人改这段代码的时候,就会强制他考虑要加的代码是否应该在方法事务内
  • 有些人可能会说,已经有了声明式事务,但是写代码的人没注意,这能怪谁
  • 话虽然是这么说,但是我们还是希望可以通过一些机制或者规范,降低这些问题发生的概率
  • 比如建议大家使用编程式事务,而不是声明式事务
  • 因为有些时候,声明式事务确实不够明显
  • 声明式事务用不对容易失效
  • 除了事务的粒度问题,还有一个问题那就是声明式事务虽然看上去帮我们简化了很多代码,但是一旦没用对,也很容易导致事务失效
  • 如以下几种场景就可能导致声明式事务失效:
  • 1、@Transactional 应用在非 public 修饰的方法上
  • 2、@Transactional 注解属性 propagation 设置错误
  • 3、@Transactional 注解属性 rollbackFor 设置错误
  • 4、同一个类中方法调用,导致@Transactional 失效
  • 5、异常被catch捕获导致@Transactional 失效
  • 6、数据库引擎不支持事务
  • 以上几个问题,如果使用编程式事务的话,很多都是可以避免的
  • 使用声明式事务失效的问题,如:
  • 因为Spring的事务是基于AOP实现的,但是在代码中,有时候我们会有很多切面,不同的切面可能会来处理不同的事情,多个切面之间可能会有相互影响
  • 在之前的一个项目中,发现Service层的事务全都失效了,一个SQL执行失败后并没有回滚,排查下来才发现,是因为新增了一个切面,这个切面里面做个异常的统一捕获,导致事务的切面没有捕获到异常,导致事务无法回滚
  • 很多人还是会说,说到底还是自己能力不行,对事务理解不透彻,用错了能怪谁
  • 但是还是那句话,我们确实无法保证所有人的能力都很高,也无法要求所有开发者都能不出错
  • 我们能做的就是,尽量可以通过机制或者规范,来避免或者降低这些问题发生的概率
  • 其实,如果大家有认真看过阿里巴巴出的那份Java开发手册的话,其实就能发现,其中的很多规约并不是完完全全容易被人理解,有些也比较生硬,但是其实,这些规范都是从无数个坑里爬出来的开发者们总结出来的
  • 关于@Transactional 的用法,规约中也有提到过,只不过规约中的观点没有这么鲜明

  • Spring事务失效可能是哪些原因
  • Spring中比较容易失效的就是通过@Transactional 定义的声明式事务,他在以下几个场景中会导致事务失效,一定要注意:
  • 1、@Transactional 应用在非 public 修饰的方法上

  • private方法,只会在当前对象中的其他方法中调用,也就是会进行对象的自调用,这种情况是用this调用的,并不会走到代理对象,而@Transactional 是基于动态代理实现的,所以代理会失效
  • 2、@Transactional 注解属性 propagation 设置错误

  • 以上,如果事务发生回滚,则methodA并不会回滚
  • 因为他的propagation是不支持事务,那么他就不会一起回滚
  • 3、@Transactional 注解属性 rollbackFor 设置错误

  • 以上,如果发生非RuntimeException,则事务不会回滚,那么就会导致事务失效
  • 所以需要指定为 (rollbackFor = Exception.class)
  • 4、同一个类中方法调用,导致@Transactional 失效

  • 以上,和private是一回事,因为没办法走到代理服务,所以事务会失效
  • 5、异常被catch捕获导致@Transactional 失效

  • 以为异常被捕获,所以就没办法基于异常进行rollback了,所以事务会失效
  • 6、数据库引擎不支持事务这个好理解,如myisam,不支持的肯定就不行了
  • @Transactional(rollbackFor = Exception.class)注解
  • Exception分为运行时异常RuntimeException和非运行时异常
  • 事务管理对于企业应用来说是至关重要的,即使出现异常情况,它也可以保证数据的一致性
  • 当 @Transactional 注解作用于类上时,该类的所有 public 方法将都具有该类型的事务属性,同时,我们也可以在方法级别使用该标注来覆盖类级别的定义
  • 如果类或者方法加了这个注解,那么这个类里面的方法抛出异常,就会回滚,数据库里面的数据也会回滚
  • 在 @Transactional 注解中如果不配置 rollbackFor 属性,那么事物只会在遇到 RuntimeException 的时候才会回滚,加上 rollbackFor=Exception.class,可以让事物在遇到非运行时异常时也回滚
  • Spring 事务的实现原理
  • Spring 事务的本质其实就是数据库对事务的支持,没有数据库的事务支持,Spring是无法提供事务功能的
  • 真正的数据库层的事务提交和回滚是通过binlog或者 redo log 实现的
  • 编程式事务管理使用 TransactionTemplate
  • 声明式事务管理是建立在 AOP 之上的
  • 其本质是通过 AOP 功能,对方法前后进行拦截,将事务处理的功能编织到拦截的方法中,也就是在目标方法开始之前加入一个事务,在执行完目标方法之后根据执行情况提交或者回滚事务
  • 声明式事务最大的优点就是不需要在业务逻辑代码中掺杂事务管理的代码,只需在配置文件中做相关的事务规则声明或通过@Transactional 注解的方式,便可以将事务规则应用到业务逻辑中
  • 声明式事务管理要优于编程式事务管理,这正是 Spring 倡导的非侵入式的开发方式,使业务代码不受污染,只要加上注解就可以获得完全的事务支持
  • 唯一不足地方是,最细粒度只能作用到方法级别,无法做到像编程式事务那样可以作用到代码块级别
  • 事务传播机制

  • 介绍
  • Spring的事务传播机制用于控制在多个事务方法相互调用时事务的行为
  • 在复杂的业务场景中,多个事务方法之间的调用可能会导致事务的不一致,如出现数据丢失、重复提交等问题,使用事务传播机制可以避免这些问题的发生,保证事务的一致性和完整性
  • Spring的事务规定了7种事务的传播级别,默认的传播机制是REQUIRED
    • required,如果不存在事务则开启一个事务,如果存在事务则加入之前的事务,总是只有一个事务在执行
    • required_new,每次执行新开一个事务
    • supported,有事务则加入事务,没有事务则普通执行
    • not_supported,有事务则暂停该事务,没有则普通执行
    • mandatory,强制有事务,没有事务则报异常
    • never,有事务则报异常
    • nested,如果之前有事务,则创建嵌套事务,嵌套事务回滚不影响父事务,反之父事务影响嵌套事务
  • Spring的事务隔离有五大隔离级别,默认值为 ISOLATION_DEFAULT(使用数据库的设置),其他四个隔离级别和数据库的隔离级别一致
  • 用法
  • 假设有两个业务方法A和B,方法A在方法B中被调用,需要在事务中保证它们的一致性,如果方法A或方法B中的任何一个方法发生异常,则需要回滚事务
  • 使用Spring的事务传播机制,可以在方法A和方法B上使用相同的事务管理器,并通过设置相同的传播行为来保证事务的一致性和完整性
  • 具体实现如下:

  • 在上述示例中,方法A和方法B都使用了REQUIRED的传播行为,表示如果当前存在事务,则在当前事务中执行;
  • 如果当前没有事务,则创建一个新的事务
  • 如果在方法A或方法B中出现异常,则整个事务会自动回滚
  • rollbackFor
  • rollbackFor是Spring事务中的一个属性,用于指定哪些异常会触发事务回滚
  • 在一个事务方法中,如果发生了rollbackFor属性指定的异常或其子类异常,则事务会回滚
  • 如果不指定rollbackFor,则默认情况下只有RuntimeException和Error会触发事务回滚
  • 场景举例
  • 问:一个长的事务方法a,在读写分离的情况下,里面既有读库操作,也有写库操作,再调用个读库方法b,方法b该用什么传播机制呢?
  • 这种情况,读方法如果是最后一步,直接not_supported就行了,避免读报错导致数据回滚
  • 如果是中间步骤,最好还是要required,因为异常失败需要回滚一下
  • 事务接口

  • PlatformTransactionManager:(平台)事务管理器
  • TransactionDefinition:事务定义信息(事务隔离级别、传播行为、超时、只读、回滚规则)
  • TransactionStatus:事务运行状态
  • 所谓事务管理,其实就是“按照给定的事务规则来执行提交或者回滚操作”
  • PlatformTransactionManager接口介绍
    • Spring并不直接管理事务,而是提供了多种事务管理器,他们将事务管理的职责委托给Hibernate或者JTA等持久化机制所提供的相关平台框架的事务来实现
    • Spring事务管理器的接口是:org.springframework.transaction.PlatformTransactionManager,通过这个接口,Spring为各个平台如JDBC、Hibernate等都提供了对应的事务管理器,但是具体的实现就是各个平台自己的事情了
  • PlatformTransactionManager接口中定义了三个方法:

  • #getTransaction(TransactionDefinition definition)方法,根据事务定义 TransactionDefinition,获得 TransactionStatus
    • 为什么不是创建事务呢?
    • 因为如果当前如果已经有事务,则不会进行创建,一般来说会跟当前线程进行绑定
    • 如果不存在事务,则进行创建
  • #commit(TransactionStatus status)方法,根据 TransactionStatus 情况,提交事务
    • 为什么根据 TransactionStatus 情况,进行提交?
    • 例如说,带@Transactional 注解的的 A方法,会调用 @Transactional 注解的的 B 方法
    • 在 B 方法结束调用后,会执行 PlatformTransactionManager#commit(TransactionStatus status)方法,此处事务是不能、也不会提交的
    • 而是在 A 方法结束调用后,执行 PlatformTransactionManager#commit(TransactionStatus status)方法,提交事务
  • 使用 Spring 事务通过 PlatformTransactionManager,为不同的数据层持久框架提供统一的API,无需关心到底是原生 JDBC、Spring JDBC、JPA、Hibernate 还是 MyBatis

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

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

相关文章

Python爬取wallhaven的所有4k壁纸图片

目录 一、前端结构 二、代码 一、前端结构 为get请求,携带参数如下: 随page参数变化网页变化 需要在整张页面中进入: //section[class"thumb-listing-page"]//li/figure/a/href 进入后获取图片src: //section[class"fit"]/div[class"s…

Hadoop伪分布式环境搭建

什么是Hadoop伪分布式集群? Hadoop 伪分布式集群是一种在单个节点上模拟分布式环境的配置,用于学习、开发和测试 Hadoop 的功能和特性。它提供了一个简化的方式来体验和熟悉 Hadoop 的各个组件,而无需配置和管理一个真正的多节点集群。 在 Ha…

国庆作业 day 1

服务器 #include<myhead.h>//消息结构体 typedef struct {long msgtype; //消息类型char data[1024]; //消息正文 }Msg_ds;#define SIZE sizeof(Msg_ds)-sizeof(long) //正文大小 int main(int argc, const char *argv[]) {//1、创建key值key_t key;…

Spring 作用域解析器AnnotationScopeMetadataResolver

博主介绍&#xff1a;✌全网粉丝近5W&#xff0c;全栈开发工程师&#xff0c;从事多年软件开发&#xff0c;在大厂呆过。持有软件中级、六级等证书。可提供微服务项目搭建与毕业项目实战&#xff0c;博主也曾写过优秀论文&#xff0c;查重率极低&#xff0c;在这方面有丰富的经…

Java项目防止SQL注入的几种方案

目录 一、什么是SQL注入&#xff1f; 二、Java项目防止SQL注入方式 1、PreparedStatement防止SQL注入 2、mybatis中#{}防止SQL注入 3、对请求参数的敏感词汇进行过滤 4、nginx反向代理防止SQL注入 一、什么是SQL注入&#xff1f; SQL注入即是指web应用程序对用户输入数…

Latent Diffusion Models 论文解读

论文下载地址&#xff1a;https://arxiv.org/pdf/2112.10752.pdf 0 摘要 通过将图像形成过程分解为去噪自编码器的连续应用&#xff0c;扩散模型&#xff08;DM&#xff09;在图像数据及其他方面取得了最先进的合成结果。此外&#xff0c;它们的表述允许一种指导机制来控制图…

HP 喷墨一体机 - “检查墨盒“指示灯闪烁,怎么办?

适用机型&#xff1a; HP PSC 1118、1218 、1318、1350、1406/1408、1508 故障现象&#xff1a; “检查墨盒”指示灯闪烁&#xff0c;“份数”显示的是英文字母“E”&#xff0c;代表 Error&#xff08;错误&#xff09;的意思。&#xff08;无复印份数显示 &#xff09; “检…

想升级macOS Big Sur,但是MacBook内存空间不够该怎么办?

随着使用时间的增长&#xff0c;我们会发现Mac电脑的存储空间越来越少&#xff0c;这时候我们就需要对Mac电脑进行清理&#xff0c;以释放更多的存储空间。那么&#xff0c;Mac空间不足怎么解决呢&#xff1f; 1.清理垃圾文件 Mac空间不足怎么解决&#xff1f;首先要做的就是清…

利用fitnesse实现api接口自动化测试

上午在园子里乱逛&#xff0c;看了不少小伙伴们分享的接口测试方面的知识&#xff0c;仔细想想&#xff0c;我做接口测试也有几个年头了&#xff0c;大家所叙述到的一些经验或多或少&#xff0c;我也曾遇到过&#xff0c;突然意识到知识的点滴积累是多么的重要&#xff0c;我记…

SRTP交叉编译与移植

1 SRTP源码下载 源码下载在github采用的库为libsrtp2.5.0: weget https://github.com/cisco/libsrtp/archive/refs/tags/v2.5.0.tar.gz2 SRTP交叉编译 新增交叉编译脚本&#xff0c;这里需要支持openssl。 ./configure --hostarm-linux-androideabi --prefix$(pwd)/object …

基于python的GUI设计

GUI设计是编程应用的一个重要载体&#xff0c;是软件开发实现的重要步骤&#xff0c;需要重要掌握&#xff0c;不管是C、JAVA、python、matlab等编程语言&#xff0c;都需要GUI界面设计来实验软件开发的前期工作&#xff0c;同时也方便按钮操作。 在设计GUI的过程中&#xff0…

ZRTP协议与原理

1 ZRTP简介 ZRTP&#xff0c;全名Z Real-time Transport Protocol&#xff0c;是一种网络协议&#xff0c;旨在为实时通信提供安全性。与其它安全协议&#xff08;如TLS和IPsec&#xff09;不同&#xff0c;ZRTP专门为实时通信设计&#xff0c;包括音频和视频通话。它是由Phil…

竞赛 深度学习 opencv python 公式识别(图像识别 机器视觉)

文章目录 0 前言1 课题说明2 效果展示3 具体实现4 关键代码实现5 算法综合效果6 最后 0 前言 &#x1f525; 优质竞赛项目系列&#xff0c;今天要分享的是 &#x1f6a9; 基于深度学习的数学公式识别算法实现 该项目较为新颖&#xff0c;适合作为竞赛课题方向&#xff0c;学…

支付环境安全漏洞介绍

1、平台支付逻辑全流程分析 2、平台支付漏洞如何利用&#xff1f;买东西还送钱&#xff1f; 3、BURP抓包分析修改支付金额&#xff0c;伪造交易状态&#xff1f; 4、修改购物车参数实现底价购买商品 5、SRC、CTF、HW项目月入10W副业之路 6、如何构建最适合自己的网安学习路线 1…

netcore Polly.Core

一、安装包 <PackageReference Include"Polly.Core" Version"8.0.0" /> 二、编写重试 ResiliencePipeline pipeline new ResiliencePipelineBuilder().AddRetry(new RetryStrategyOptions{MaxRetryAttempts 2,Delay TimeSpan.FromSeconds(3),O…

竞赛 机器视觉目标检测 - opencv 深度学习

文章目录 0 前言2 目标检测概念3 目标分类、定位、检测示例4 传统目标检测5 两类目标检测算法5.1 相关研究5.1.1 选择性搜索5.1.2 OverFeat 5.2 基于区域提名的方法5.2.1 R-CNN5.2.2 SPP-net5.2.3 Fast R-CNN 5.3 端到端的方法YOLOSSD 6 人体检测结果7 最后 0 前言 &#x1f5…

Flink的处理函数——processFunction

目录 一、处理函数概述 二、Process函数分类——8个 &#xff08;1&#xff09;ProcessFunction &#xff08;2&#xff09;KeyedProcessFunction &#xff08;3&#xff09;ProcessWindowFunction &#xff08;4&#xff09;ProcessAllWindowFunction &#xff…

CSDN Markdown

这里写自定义目录标题 欢迎使用Markdown编辑器新的改变功能快捷键合理的创建标题&#xff0c;有助于目录的生成如何改变文本的样式插入链接与图片如何插入一段漂亮的代码片生成一个适合你的列表创建一个表格设定内容居中、居左、居右SmartyPants 创建一个自定义列表如何创建一个…

数据中台实战(05)-如何统一管理纷繁杂乱的数据指标?

各种类型的元数据有什么用&#xff1f;跟数据中台啥关系&#xff1f; 元数据在指标管理、模型设计、数据质量和成本治理四个领域都发挥作用&#xff0c;这些领域构成数据中台OneData 数据体系。今天逐一了解元数据在上述领域的应用 1 指标管理 指标&#xff0c;一种特定类型…

【Java项目推荐之黑马头条】你的发布文章业务是怎么实现的?

前言 在学习Java的路上还是遇到了很多不错的好项目的&#xff0c;今天分享给大家&#xff0c;希望能对大家面试有所帮助&#xff01; 后续会继续推荐其他好的项目&#xff0c;这次推荐的是B站开源的视频黑马头条项目&#xff0c;来吧学会它一起去虐面试官&#xff01;&#x…