Spring 事务原理总结六

不知不觉,关于Spring事务的文章已经写了五篇了。老实讲我自己不断质疑过自己:现在写这些文章还有意义吗?当前的市场已经成什么样了,为什么还要固守这落后的技术?但是贝索斯一次接受访谈的回答,让我写下去的决心更加坚定了,他是这样说的:相较于千变万化的事物,我更关注那些恒久不变的东西!

书归正传,上篇文章我们解决了《Spring事务原理总结四》这篇文章中提到的几个问题中的三个,其中“Spring事务异常回滚执行流程”这个问题,我们并没有梳理。今天就借这篇文章详细梳理一下。如果各位觉得这些文章对您有用,还请多多关注,谢谢!如果大家觉得有哪些地方梳理的不正确,也请大家多多指教,非常感谢!本篇文章梳理的比较啰嗦,大家可以跳过中间过程,看最后的总结。如果有些地方大家觉得不对,欢迎指出。

执行流程梳理

继续采用《Spring事务原理总结一》中的案例,修改TransferServiceImpl类中的check(String, String, BigDecimal)方法,具体代码如下所示:

@Override
public void check(String from, String to, BigDecimal money) {System.out.println("校验开始");System.out.println("校验中||...........");try {System.out.println(1 / 0);Thread.sleep(1000 * 5);} catch (InterruptedException e) {}System.out.println("校验中==...........");System.out.println("校验结束");}

启动程序(用debug模式运行SpringTransactionApplication类即可),接着会在TransactionAspectSupport类的invokeWithinTransaction(Method method, @Nullable Class<?> targetClass, final InvocationCallback invocation)方法中的断点处停下,具体如下图所示:

启动程序(用debug模式运行SpringTransactionApplication类即可),接着会在TransactionAspectSupport类的invokeWithinTransaction(Method method, @Nullable Class<?> targetClass, final InvocationCallback invocation)方法中的断点处停下,具体如下图所示:

接着让我们继续执行代码,直到程序执行到下图所示的断点处停下,详情情况请参见下面这幅图:

这里需要注意一下,控制台并未输出TransferServiceImpl#check(String, String, BigDecimal)方法中要打印的任何内容,然后继续执行,结果如下图所示:

由图中可以看出,控制台输出了TransferServiceImpl#check(String, String, BigDecimal)方法要打印的内容,并且程序直接进入了catch逻辑。仔细观察会发现这个异常类型为java.lang.ArithmeticException: / by zero,这就是我们在代码中添加的System.out.println(1 / 0)抛出的,如果这个方法实际操作的是数据库,那catch逻辑中要执行的就是回滚操作了。先来看一下这个方法(这个方法——completeTransactionAfterThrowing(TransactionInfo, Throwable)——位于TransactionAspectSupport类中)的源码,如下所示:

protected void completeTransactionAfterThrowing(@Nullable TransactionInfo txInfo, Throwable ex) {if (txInfo != null && txInfo.getTransactionStatus() != null) {if (logger.isTraceEnabled()) {logger.trace("Completing transaction for [" + txInfo.getJoinpointIdentification() +"] after exception: " + ex);}if (txInfo.transactionAttribute != null && txInfo.transactionAttribute.rollbackOn(ex)) {try {txInfo.getTransactionManager().rollback(txInfo.getTransactionStatus());}catch (TransactionSystemException ex2) {logger.error("Application exception overridden by rollback exception", ex);ex2.initApplicationException(ex);throw ex2;}catch (RuntimeException | Error ex2) {logger.error("Application exception overridden by rollback exception", ex);throw ex2;}}else {// We don't roll back on this exception.// Will still roll back if TransactionStatus.isRollbackOnly() is true.try {txInfo.getTransactionManager().commit(txInfo.getTransactionStatus());}catch (TransactionSystemException ex2) {logger.error("Application exception overridden by commit exception", ex);ex2.initApplicationException(ex);throw ex2;}catch (RuntimeException | Error ex2) {logger.error("Application exception overridden by commit exception", ex);throw ex2;}}}
}

先来看一下下面这幅运行时图片,程序运行到if (txInfo.transactionAttribute != null && txInfo.transactionAttribute.rollbackOn(ex))处停下,具体见下图:

先让我们一起看一下txInfo.transactionAttribute.rollbackOn(ex)这句,这段代码的详细执行流程为:

  1. 执行DelegatingTransactionAttribute#rollbackOn(Throwable ex),注意DelegatingTransaction-Attribute中持有一个TransactionAttribute对象
  2. 调用RuleBasedTransactionAttribute#rollbackOn(Throwable ex)
  3. 调用DefaultTransactionAttribute#rollbackOn(Throwable ex)

对于这个调用过程,我们需要注意以下几点:

  • DelegatingTransactionAttribute类中持有的TransactionAttribute对象的实际类型是RuleBaseTransactionAttribute,这个类的rollbackOn(Throwable)方法实际上是一个代理方法,其会把处理转发给RuleBaseTransactionAttribute这个实际类型中的rollbackOn(Throwable)方法。所以DelegatingTransactionAttribute类中的rollbackOn(Throwable)方法的源码非常简单,具体如下所示:
public boolean rollbackOn(Throwable ex) {return this.targetAttribute.rollbackOn(ex);
}
  • RuleBasedTransactionAttribute这个类的rollbackOn(Throwable)方法是实际进行判断的地方,其中有一个rollbackRules对象,该对象的类型是List,其中存储的是一个一个的RollbackRuleAttribute类型的对象,首先来看一下rollbackOn(Throwable)这个方法的源码吧:
public boolean rollbackOn(Throwable ex) {RollbackRuleAttribute winner = null;int deepest = Integer.MAX_VALUE;if (this.rollbackRules != null) {for (RollbackRuleAttribute rule : this.rollbackRules) {int depth = rule.getDepth(ex);if (depth >= 0 && depth < deepest) {deepest = depth;winner = rule;}}}// User superclass behavior (rollback on unchecked) if no rule matches.if (winner == null) {return super.rollbackOn(ex);}return !(winner instanceof NoRollbackRuleAttribute);
}

通过这段源码不难发现,程序首先会遍历本类持有的rollbackRules对象,从中找到适合的数据并赋值给winner;接着判断winner对象是否为空,如果winner对象为空,则直接调用父类的rollbackOn(Throwable)方法,判断当前的异常类型是否合法,否则判断当前的winner对象是否为NoRollbackRuleAttribute类型,并取反,然后将结果返回给上级调用者。下面让我们看一下RollbackRuleAttribute类的继承结构,具体如下图所示:

接下来让我们看一下RuleBasedTransactionAttribute的父类DefaultTransactionAttribute类中的rollbackOn(Throwable)方法的源码

@Override
public boolean rollbackOn(Throwable ex) {return (ex instanceof RuntimeException || ex instanceof Error);
}

这个方法的主要作用就是判断当前的异常类型是否为RuntimeException或Error,如果是这两个就返回true,否则就返回false

看到这里我不禁有个问题:RuleBasedTransactionAttribute对象中的rollbackRules对象中的值是从哪里来的?接着让我们改造以下TransferServiceImpl类上的@Transactional注解,在其中增加一个rollbackFor属性,具体如下图所示:

然后启动程序,查看结果(注意断点在RuleBasedTransactionAttribute#rollbackOn()方法中),具体如下图所示:

从图中不难发现此时rollbackRules对象中的值确实是我们通过rollbackFor属性注入的两个异常类。但我又有点好奇RuleBasedTransactionAttribute#rollbackOn(Throwable)方法中的最后一句“!(winner instanceof NoRollbackRuleAttribute)”中的NoRollbackRuleAttribute这个是从哪里来的?还记得@Transactional注解上的noRollbackFor属性吗?让我们继续改造TransferServiceImpl类上的@Transactional注解,在其上添加noRollbackFor属性,具体如下图所示:

然后重新启动程序,查看结果(注意断点在RuleBasedTransactionAttribute#rollbackOn()方法中),具体如下图所示:

从图中可以看出,我们在注解中指定的CustomException被包装成了NoRollbackRuleAttribute类型的对象,由于前面我们更改了TransferServiceImpl类的check()方法,其最终会抛出一个名为CustomException类型的异常,所以这段代码返回的结果是false,最终也就不会执行TransactionAspectSupport中的completeTransactionAfterThrowing(TransactionInfo, Throwable)方法。注意TransferServiceImpl类中的check()方法的源码为:

public void check(String from, String to, BigDecimal money) throws Exception {System.out.println("校验开始");System.out.println("校验中||...........");try {// System.out.println(1 / 0);Thread.sleep(1000 * 5);throw new CustomException();} catch (InterruptedException e) {}System.out.println("校验中==...........");System.out.println("校验结束");

抛开这些问题,继续回到DefaultTransactionAttribute#rollbackFor()方法中,最终程序抛出的ArithmeticException异常经过该方法后返回的结果为true。具体如下图所示:

总体来看在不指定@Transactional注解的rollbackFor属性的时候,调用RuleBasedTransact-ionAttribute#rollbackOn()方法的作用就是判断当前异常是否为运行时异常(即RuntimeException或者Error),如果是则触发后面的回滚逻辑,如果不是则不触发后面的回滚逻辑

接下来让我们继续回到TransactionAspectSupport#completeTransactionAfterThrowing(Tra-nsactionInfo, Throwable)方法中,经过前面的判断,最终代码走到了txInfo.getTransactionManag-er().rollback(txInfo.getTransactionStatus())这行,具体如下图所示:

首先来看一下TransactionManager类的rollback(TransactionStatus)方法的源码(实际代码位于AbstractPlatformTransactionManager类中):

public final void rollback(TransactionStatus status) throws TransactionException {if (status.isCompleted()) {throw new IllegalTransactionStateException("Transaction is already completed - do not call commit or rollback more than once per transaction");}DefaultTransactionStatus defStatus = (DefaultTransactionStatus) status;processRollback(defStatus, false);
}

接着再来看一下processRollback()方法的源码,其主要作用就是执行真正的回滚逻辑。下面一起看一下processRollback()方法的源码:

private void processRollback(DefaultTransactionStatus status, boolean unexpected) {try {boolean unexpectedRollback = unexpected;boolean rollbackListenerInvoked = false;try {triggerBeforeCompletion(status);if (status.hasSavepoint()) {if (status.isDebug()) {logger.debug("Rolling back transaction to savepoint");}this.transactionExecutionListeners.forEach(listener -> listener.beforeRollback(status));rollbackListenerInvoked = true;status.rollbackToHeldSavepoint();}else if (status.isNewTransaction()) {if (status.isDebug()) {logger.debug("Initiating transaction rollback");}this.transactionExecutionListeners.forEach(listener -> listener.beforeRollback(status));rollbackListenerInvoked = true;doRollback(status);}else {// Participating in larger transactionif (status.hasTransaction()) {if (status.isLocalRollbackOnly() || isGlobalRollbackOnParticipationFailure()) {if (status.isDebug()) {logger.debug("Participating transaction failed - marking existing transaction as rollback-only");}doSetRollbackOnly(status);}else {if (status.isDebug()) {logger.debug("Participating transaction failed - letting transaction originator decide on rollback");}}}else {logger.debug("Should roll back transaction but cannot - no transaction available");}// Unexpected rollback only matters here if we're asked to fail earlyif (!isFailEarlyOnGlobalRollbackOnly()) {unexpectedRollback = false;}}}catch (RuntimeException | Error ex) {triggerAfterCompletion(status, TransactionSynchronization.STATUS_UNKNOWN);if (rollbackListenerInvoked) {this.transactionExecutionListeners.forEach(listener -> listener.afterRollback(status, ex));}throw ex;}triggerAfterCompletion(status, TransactionSynchronization.STATUS_ROLLED_BACK);if (rollbackListenerInvoked) {this.transactionExecutionListeners.forEach(listener -> listener.afterRollback(status, null));}// Raise UnexpectedRollbackException if we had a global rollback-only markerif (unexpectedRollback) {throw new UnexpectedRollbackException("Transaction rolled back because it has been marked as rollback-only");}}finally {cleanupAfterCompletion(status);}
}

通过源码不难发现该方法接收一个TransactionStatus对象作为参数,这个对象封装了当前事务的状态信息。当需要回滚事务时(例如遇到未捕获异常或者显式调用TransactionTemplate或PlatformTransactionManager的rollback()方法时),框架会调用此方法来完成以下任务

  1. 清理资源:根据事务的具体类型(如JDBC、Hibernate、JTA等)清理与事务相关的一些资源,这可能包括数据库连接的回滚操作或者其他事务性资源的相应清理工作
  2. 更新事务状态:将事务状态标记为已回滚,确保后续不会尝试提交这个事务
  3. 触发监听器或回调:如果有注册的事务同步监听器(TransactionSynchronizationAdapter),则会触发相应的 afterCompletion 回调方法,通知它们事务已经回滚

总之,processRollback方法实现了事务生命周期中的“回滚”阶段,确保事务能够按照预期进行回滚,从而维持事务的原子性和一致性。通过debug跟踪和阅读源码,我们发现本示例最终走到了elsle if分支,该逻辑片段最终会调用DatasourceTransactionManager类的doRollback(DefaultTransactionStatus)方法,先来看一下这个方法的源码吧,具体如下所示:

protected void doRollback(DefaultTransactionStatus status) {DataSourceTransactionObject txObject = (DataSourceTransactionObject) status.getTransaction();Connection con = txObject.getConnectionHolder().getConnection();if (status.isDebug()) {logger.debug("Rolling back JDBC transaction on Connection [" + con + "]");}try {con.rollback();}catch (SQLException ex) {throw translateException("JDBC rollback", ex);}
}

通过源码,我们可以很清晰的看到,其最终就是通过调用Connection对象上的rollback()方法来完成事务的回滚的。

总结

文章进行到这里,前面遗留的问题基本上就梳理完了。首先通过这篇文章我们可以很清晰的看到Spring的设计者利用动态代理(cglib动态代理)及各种设计模式(责任链、模板等设计模式,其中模板设计模式在AbstractPlatformTransactionManager类体现的最为明显。该类中的rollback(TransactionStatus)方法规定了调用流程,即调用本类的processRollback(DefaultTransactionStatus, boolean)方法,而该方法又继续调用了本类中的模板方法doRollback(DefaultTransactionStatus),实际上调用的是其实现类中的方法完成了事务逻辑的抽离,使开发者可以花费更多的精力在业务代码的编写上。下面就让我们一起看一下AbstractPlatformTransactionManager类中的这几个方法的源码:

public final void rollback(TransactionStatus status) throws TransactionException {if (status.isCompleted()) {throw new IllegalTransactionStateException("Transaction is already completed - do not call commit or rollback more than once per transaction");}DefaultTransactionStatus defStatus = (DefaultTransactionStatus) status;processRollback(defStatus, false);
}
// 
private void processRollback(DefaultTransactionStatus status, boolean unexpected) {try {boolean unexpectedRollback = unexpected;boolean rollbackListenerInvoked = false;try {triggerBeforeCompletion(status);if (status.hasSavepoint()) {if (status.isDebug()) {logger.debug("Rolling back transaction to savepoint");}this.transactionExecutionListeners.forEach(listener -> listener.beforeRollback(status));rollbackListenerInvoked = true;status.rollbackToHeldSavepoint();}else if (status.isNewTransaction()) {if (status.isDebug()) {logger.debug("Initiating transaction rollback");}this.transactionExecutionListeners.forEach(listener -> listener.beforeRollback(status));rollbackListenerInvoked = true;doRollback(status);}else {// Participating in larger transactionif (status.hasTransaction()) {if (status.isLocalRollbackOnly() || isGlobalRollbackOnParticipationFailure()) {if (status.isDebug()) {logger.debug("Participating transaction failed - marking existing transaction as rollback-only");}doSetRollbackOnly(status);}else {if (status.isDebug()) {logger.debug("Participating transaction failed - letting transaction originator decide on rollback");}}}else {logger.debug("Should roll back transaction but cannot - no transaction available");}// Unexpected rollback only matters here if we're asked to fail earlyif (!isFailEarlyOnGlobalRollbackOnly()) {unexpectedRollback = false;}}}catch (RuntimeException | Error ex) {triggerAfterCompletion(status, TransactionSynchronization.STATUS_UNKNOWN);if (rollbackListenerInvoked) {this.transactionExecutionListeners.forEach(listener -> listener.afterRollback(status, ex));}throw ex;}triggerAfterCompletion(status, TransactionSynchronization.STATUS_ROLLED_BACK);if (rollbackListenerInvoked) {this.transactionExecutionListeners.forEach(listener -> listener.afterRollback(status, null));}// Raise UnexpectedRollbackException if we had a global rollback-only markerif (unexpectedRollback) {throw new UnexpectedRollbackException("Transaction rolled back because it has been marked as rollback-only");}}finally {cleanupAfterCompletion(status);}
}
// 
protected abstract void doRollback(DefaultTransactionStatus status) throws TransactionException;

下面再回顾一下AbstractPlatformTransactionManager的实现类DataSourceTransactionManager中的doRollback()方法的源码(如果向了解这两个类的继承关系,可以浏览《Spring 事务原理总结三》这篇文章)

protected void doRollback(DefaultTransactionStatus status) {DataSourceTransactionObject txObject = (DataSourceTransactionObject) status.getTransaction();Connection con = txObject.getConnectionHolder().getConnection();if (status.isDebug()) {logger.debug("Rolling back JDBC transaction on Connection [" + con + "]");}try {con.rollback();}catch (SQLException ex) {throw translateException("JDBC rollback", ex);}
}

接着通过本章强化了我们对@Transactional注解的认识,该注解上的rollbackFor属性可以指定哪些异常发生时需要执行回滚操作,该注解上的noRollbackFor属性则可以指定哪些异常发生时不需要执行回滚操作(Spring设计者的这个设计,为开发者在开发中按照业务异常分别处理异常,提供了很大的便利)。另外通过本章我们也知道了通过这两个属性指定的异常最终会被分别包装成RollbackRuleAttribute和No RollbackRuleAttribute类型的对象,程序执行时会通过RuleBasedTransactionAttribute中的rollbackOn(Throwable ex)方法进行区分并加以判断,具体判断代码如下所示(含DelegatingTransactionAttribute和RuleBasedTransactionAttribute的父类DefaultTransactionAttribute中的rollbackOn(Throwable)方法):

/ DelegatingTransactionAttribute类中的rollbackOn(Throwable)方法
public boolean rollbackOn(Throwable ex) {return this.targetAttribute.rollbackOn(ex);
}
// RuleBasedTransactionAttribute类中的rollbackOn(Throwable)方法
public boolean rollbackOn(Throwable ex) {RollbackRuleAttribute winner = null;int deepest = Integer.MAX_VALUE;if (this.rollbackRules != null) {for (RollbackRuleAttribute rule : this.rollbackRules) {int depth = rule.getDepth(ex);if (depth >= 0 && depth < deepest) {deepest = depth;winner = rule;}}}// User superclass behavior (rollback on unchecked) if no rule matches.if (winner == null) {return super.rollbackOn(ex);}return !(winner instanceof NoRollbackRuleAttribute);
}
// RuleBasedTransactionAttribute的父类DefaultTransactionAttribute中的rollbackOn(Throwable)方法
@Override
public boolean rollbackOn(Throwable ex) {return (ex instanceof RuntimeException || ex instanceof Error);
}

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

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

相关文章

C++多重继承

C多重继承 C中的多重继承是指一个类可以从多于一个的基类派生出来&#xff0c;这允许在一个派生类中继承多个基类的特性和行为。多重继承增加了C的灵活性和表达能力&#xff0c;但同时也带来了一些复杂性&#xff0c;如菱形继承问题和潜在的命名冲突。 基本用法 定义一个多重…

【Spring篇】 项目加盐加密处理

目录 1. MD5 加密算法 2. 加盐加密流程 3. Spring Security 实现加盐加密 1. 添加 Spring Security 框架 2. 关闭 Spring Security 认证 3.实现加盐加密 1. MD5 加密算法 MD5 是 Message Digest Algorithm 的缩写&#xff0c;译为信息摘要算法&#xff0c;它是 Java …

电梯控制系列之电梯结构介绍

这篇博客介绍单部10层电梯的完整控制程序框架编写过程&#xff0c;编程语言&#xff1a;SCL&#xff0c;控制器型号&#xff1a;S7-1200PLC。本篇博客介绍和电梯控制相关的一些电梯结构介绍。本文只可作为学习参考资料&#xff0c;行业控制需要遵循电梯安全相关规范。 1、电梯…

线程和进程【并发和并行、线程上下文切换、线程的状态】

线程和进程【并发和并行、线程上下文切换、线程的状态】 什么是并发与并行&#xff1f;什么是线程上下文切换&#xff1f;线程状态&#xff1a;一个线程的一生 转自 极客时间 进程&#xff1a;是指内存中运行的一个应用程序&#xff0c;每个进程都有自己独立的内存空间&#x…

JavaSE-03笔记【继承~super】

文章目录 1. 继承1.1 继承概述&#xff08;理解&#xff09;1.2 如何继承&#xff08;掌握&#xff09;1.2.1 继承的语法格式1.2.2 具体举例 1.3 继承的相关特性&#xff08;掌握&#xff09;1.4 对继承自Object类的方法的测试&#xff08;理解&#xff09;1.5 难点解惑1.5.1 掌…

Switch开关(antd-design组件库)简单使用

1.Switch开关 开关选择器。 2.何时使用 需要表示开关状态/两种状态之间的切换时&#xff1b; 和 checkbox 的区别是&#xff0c;切换 switch 会直接触发状态改变&#xff0c;而 checkbox 一般用于状态标记&#xff0c;需要和提交操作配合。 组件代码来自&#xff1a; 开关 Swit…

【力扣】169.多数元素

这道题的解法是运用哈希表打擂台的思想 首先题目的意思是存在数字&#xff0c;意思就是最后返回的结果不可能为空就是了&#xff0c;所以便不用考虑{1&#xff0c;2&#xff0c;3&#xff0c;4&#xff0c;5}这种例子。那么就可以用哈希表存所出现数字出现的次数&#xff0c;然…

11-k8s中网络资源service

一、service资源概述 每当我们企业的业务pod迭代功能的时候&#xff0c;都会修改pod&#xff0c;修改后重新启动pod&#xff0c;ip就会变化&#xff0c;那么在生产环境当中&#xff0c;从用户到宿主机、从宿主机到pod&#xff0c;这一个访问流程&#xff0c;都是事先写好的&…

MySQL数据库基础(一):数据库概述

文章目录 数据库概述 一、数据库介绍 二、数据库分类 1、关系型数据库 2、非关系型数据库NoSQL 三、常见数据库介绍 1、关系型数据库 2、非关系型数据库 数据库概述 一、数据库介绍 数据库就是存储数据的仓库&#xff0c;其本质是一个文件系统&#xff0c;按照特定的…

租用一个服务器需要多少钱?2024阿里云新版报价

2024年最新阿里云服务器租用费用优惠价格表&#xff0c;轻量2核2G3M带宽轻量服务器一年61元&#xff0c;折合5元1个月&#xff0c;新老用户同享99元一年服务器&#xff0c;2核4G5M服务器ECS优惠价199元一年&#xff0c;2核4G4M轻量服务器165元一年&#xff0c;2核4G服务器30元3…

小白必看,总结前端所有主流的构建工具,webpack / vite / roollup / esbuild,包含源码,建议关注+收藏

前言 本篇文章旨在总结前端常见的构建工具&#xff0c;构建工具是前端工程化中的重要的组成部分。 在实际项目中&#xff0c;我们初始化项目&#xff0c;一般是使用脚手架命令一键生成的&#xff0c;比如说使用 create-vue 初始化 vue 项目的时候&#xff0c;就会默认使用 vi…

树莓派:使用mdadm为重要数据做RAID 1保护

树莓派作为个人服务器可玩性还是有点的。说到服务器&#xff0c;在企业的生成环境中为了保护数据&#xff0c;基本上都会用到RAID技术。比如&#xff0c;服务器两块小容量但高性能的盘做个RAID-1按装操作系统&#xff0c;余下的大容量中性能磁盘做个RAID-5或者RAID-6存放数据。…