博客
关于我
Spring源码分析十八:事务实现④ - 事务的回滚
阅读量:503 次
发布时间:2019-03-07

本文共 5289 字,大约阅读时间需要 17 分钟。

事务的回滚机制解析

在Spring事务管理中,回滚机制是一个核心的功能,用于确保事务能够在出现异常时正确进行 rollback。当方法抛出异常时,Spring会调用TransactionAspectSupport#completeTransactionAfterThrowing方法来处理异常,并决定是否进行事务回滚。本文将深入分析事务回滚的逻辑和实现细节。

文章目录

  • 1. 回滚触发机制

    • 1.1 形成事务的必要条件

      • 事务信息不为 null
      • 事务状态不为 completed

在Spring中,事务的回滚主要由以下几部分组成:

1. 回滚触发机制

当方法抛出异常时,Spring会判断是否需要进行事务回滚的条件具体是:

if (txInfo != null && txInfo.getTransactionStatus() != null)

这表示当前线程是否存在一个有效的事务信息。当满足条件时,进一步检查事务属性中的rollbackOn方法来决定是否进行回滚。

默认实现如下:

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

这意味着,只有当异常是RuntimeExceptionError类型时,才会触发回滚操作。其他类型的异常则不会触发回滚,但这并不意味着事务没有被启用。

2. 回滚操作的具体实现

当决定进行回滚时,Spring会调用底层平台事务管理器的rollback方法来执行回滚操作。具体实现涉及以下几个步骤:

2.1 触发回滚前置操作

在回滚前,Spring会触发注册的TransactionSynchronizationbeforeCompletion方法。这类触发器可以用于执行自定义操作,如日志记录或资源释放等。默认的实现代码如下:

protected final void triggerBeforeCompletion(DefaultTransactionStatus status) { if (status.isNewSynchronization()) { if (status.isDebug()) { logger.trace("Triggering beforeCompletion synchronization"); } TransactionSynchronizationUtils.triggerBeforeCompletion(); }}

这是一个典型的回滚前置处理逻辑。

2.2 实际的回滚逻辑处理

回滚逻辑的核心在于调用底层数据库连接的rollback方法。不过,在具体实现中,Spring使用了三种不同的回滚模式:

2.2.1 状态保存点回滚

当事务存在已保存的状态点时,采用rollbackToHeldSavepoint方法进行回滚。这通常用于嵌入式事务中,以防止外部事务受到内嵌事务异常的影响。代码实现如下:

public void rollbackToHeldSavepoint() throws TransactionException { Object savepoint = getSavepoint(); if (savepoint == null) { throw new TransactionUsageException( "Cannot roll back to savepoint - no savepoint associated with current transaction"); } getSavepointManager().rollbackToSavepoint(savepoint); getSavepointManager().releaseSavepoint(savepoint); setSavepoint(null);}

2.2.2 独立事务回滚

对于独立的事务(没有保存点),Spring直接调用底层数据源的rollback方法。具体实现如下:

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

这里直接调用数据库连接的rollback方法来完成事务回滚。

2.2.3 大事务中的回滚标识

在外部事务中,当内嵌事务设置rollbackOnly为true时,遵循JTA规范,只标记为回滚,等待外部事务统一处理。具体实现如下:

protected void doSetRollbackOnly(DefaultTransactionStatus status) { DataSourceTransactionObject txObject = (DataSourceTransactionObject) status.getTransaction(); if (status.isDebug()) { logger.debug("Setting JDBC transaction [" + txObject.getConnectionHolder().getConnection() + ] "rollback-only"); } txObject.setRollbackOnly();}

2.3 回滚后的清理工作

在完成回滚操作后,Spring需要清理事务相关资源。这主要包括:

2.3.1 设置完成标记

cleanupAfterCompletion方法进行清理。这个方法主要负责:

private void cleanupAfterCompletion(DefaultTransactionStatus status) { status.setCompleted(); if (status.isNewSynchronization()) { // 清除当前线程的中关于该事务的信息 TransactionSynchronizationManager.clear(); } if (status.isNewTransaction()) { // 清理事务信息 doCleanupAfterCompletion(status.getTransaction()); } if (status.getSuspendedResources() != null) { if (status.isDebug()) { logger.debug("Resuming suspended transaction after completion of inner transaction"); } Object transaction = (status.hasTransaction() ? status.getTransaction() : null); resume(transaction, (SuspendedResourcesHolder) status.getSuspendedResources()); }}

其中,resume方法负责恢复挂起的事务。

2.3.2 恢复挂起的事务

如果在回滚过程中有挂起的事务,Spring会在回滚完成后自动恢复该事务。代码如下:

protected final void resume(@Nullable Object transaction, @Nullable SuspendedResourcesHolder resourcesHolder) throws TransactionException { if (resourcesHolder != null) { // 恢复挂起的事务 Object suspendedResources = resourcesHolder.suspendedResources; if (suspendedResources != null) { doResume(transaction, suspendedResources); } List
suspendedSynchronizations = resourcesHolder.suspendedSynchronizations; // 设置事务相关属性 if (suspendedSynchronizations != null) { TransactionSynchronizationManager.setActualTransactionActive(resourcesHolder.wasActive); TransactionSynchronizationManager.setCurrentTransactionIsolationLevel(resourcesHolder.isolationLevel); TransactionSynchronizationManager.setCurrentTransactionReadOnly(resourcesHolder.readOnly); TransactionSynchronizationManager.setCurrentTransactionName(resourcesHolder.name); doResumeSynchronization(suspendedSynchronizations); } }}

这里的doResume方法负责具体执行事务恢复操作。

通过以上分析可以看出,Spring的事务回滚机制虽然负责核心逻辑,但实际的数据库操作由底层 JDBC 连接管理器完成。这种设计使得 Spring 在事务管理方面保持了高度的灵活性和兼容性。理解这一点对于深入研究Spring的事务管理是非常重要的。

相关标签:Sprin事务管理,TransactionAspectSupport,回滚机制,JDBC事务管理,Spring源码解析

关键词:事务回滚,Spring事务管理,回滚机制,JDBC事务,事务异常处理

转载地址:http://zykcz.baihongyu.com/

你可能感兴趣的文章
NIO笔记---上
查看>>
NIO蔚来 面试——IP地址你了解多少?
查看>>
NISP一级,NISP二级报考说明,零基础入门到精通,收藏这篇就够了
查看>>
NISP国家信息安全水平考试,收藏这一篇就够了
查看>>
NIS服务器的配置过程
查看>>
Nitrux 3.8 发布!性能全面提升,带来非凡体验
查看>>
NiuShop开源商城系统 SQL注入漏洞复现
查看>>
NI笔试——大数加法
查看>>
NLog 自定义字段 写入 oracle
查看>>
NLog类库使用探索——详解配置
查看>>
NLP 基于kashgari和BERT实现中文命名实体识别(NER)
查看>>
NLP 模型中的偏差和公平性检测
查看>>
Vue3.0 性能提升主要是通过哪几方面体现的?
查看>>
NLP 项目:维基百科文章爬虫和分类【01】 - 语料库阅读器
查看>>
NLP_什么是统计语言模型_条件概率的链式法则_n元统计语言模型_马尔科夫链_数据稀疏(出现了词库中没有的词)_统计语言模型的平滑策略---人工智能工作笔记0035
查看>>
NLP三大特征抽取器:CNN、RNN与Transformer全面解析
查看>>
NLP学习笔记:使用 Python 进行NLTK
查看>>
NLP度量指标BELU真的完美么?
查看>>
NLP的不同研究领域和最新发展的概述
查看>>
NLP的神经网络训练的新模式
查看>>