广告
返回顶部
首页 > 资讯 > 精选 >spring的事务传播属性REQUIRED_NESTED的原理介绍
  • 334
分享到

spring的事务传播属性REQUIRED_NESTED的原理介绍

2023-07-06 14:07:39 334人浏览 八月长安
摘要

这篇文章主要介绍了spring的事务传播属性REQUIRED_NESTED的原理介绍,具有一定借鉴价值,需要的朋友可以参考下。下面就和我一起来看看吧。传统事务中回滚点的使用package com.morris.spring.dem

这篇文章主要介绍了spring事务传播属性REQUIRED_NESTED的原理介绍,具有一定借鉴价值,需要的朋友可以参考下。下面就和我一起来看看吧。

传统事务中回滚点的使用

package com.morris.spring.demo.jdbc;import java.sql.*;public class TraditionSavePointDemo {public static void main(String[] args) throws SQLException {String url = "jdbc:mysql://127.0.0.1:3306/test?useUnicode=true&allowMultiQueries=true&characterEncoding=UTF-8&useFastDateParsing=false&zeroDateTimeBehavior=convertToNull";String username = "user";String passWord = "user";Connection connection = DriverManager.getConnection(url, username, password);connection.setAutoCommit(false); // 不自动提交Savepoint one = connection.setSavepoint("one");Savepoint two = null;try {Statement statement = connection.createStatement();statement.execute("insert into t_Good(good_name, price) values('iphone14', 9999)");statement.close();two = connection.setSavepoint("two");} catch (Exception e) {e.printStackTrace();connection.rollback(one); // 回滚事务}try {Statement statement = connection.createStatement();statement.execute("insert into t_good(good_name, price) values('iphone15', 9999)");statement.close();boolean flag = true;if(flag) {throw new RuntimeException("xxxx");}} catch (Exception e) {e.printStackTrace();connection.rollback(two); // 回滚事务}connection.commit();}}

在一个事务中可以指定回滚事务到某一个阶段,实现精确控制事务。

事务的传播属性NESTED

在spring中,要想使用事务中的回滚点,可以使用传播属性NESTED。

com.morris.spring.service.TransactionService#addGoodAndArea

@Transactional(propagation = Propagation.REQUIRED)public void addGoodAndArea() {System.out.println("------addGoodAndArea-------");goodService.addGood();areaService.addArea(0);}

com.morris.spring.service.AreaServiceImpl#addArea

@Transactional(propagation = Propagation.NESTED)@Overridepublic boolean addArea(int i) {int y = 1000000 / i;Area area = new Area();area.setAreaCode(y);area.setAreaName("shenzhen");return areaDao.insert(area);}

com.morris.spring.service.GoodServiceImpl#addGood

@Transactional(propagation = Propagation.NESTED)@Overridepublic boolean addGood() {Good good = new Good();good.setGoodName("iphone");good.setPrice(BigDecimal.valueOf(99999));return goodDao.insert(good);}

运行结果如下:

DEBUG DataSourceTransactionManager:384 - Creating new transaction with name [com.morris.spring.service.TransactionService.addGoodAndArea]: PROPAGATION_REQUIRED,ISOLATION_DEFAULTDEBUG DriverManagerDataSource:144 - Creating new JDBC DriverManager Connection to [jdbc:Mysql://127.0.0.1:3306/test?useUnicode=true&allowMultiQueries=true&characterEncoding=UTF-8&useFastDateParsing=false&zeroDateTimeBehavior=convertToNull]DEBUG DataSourceTransactionManager:267 - Acquired Connection [com.mysql.cj.jdbc.ConnectionImpl@8ef162] for JDBC transactionDEBUG DataSourceTransactionManager:285 - Switching JDBC Connection [com.mysql.cj.jdbc.ConnectionImpl@8ef162] to manual commit------addGoodAndArea-------DEBUG DataSourceTransactionManager:477 - Creating nested transaction with name [com.morris.spring.service.GoodServiceImpl.addGood]DEBUG JdbcTemplate:860 - Executing prepared SQL updateDEBUG JdbcTemplate:609 - Executing prepared SQL statement [insert into t_good(good_name, price) values(?,?)]DEBUG DataSourceTransactionManager:767 - Releasing transaction savepointDEBUG DataSourceTransactionManager:477 - Creating nested transaction with name [com.morris.spring.service.AreaServiceImpl.addArea]DEBUG DataSourceTransactionManager:870 - Rolling back transaction to savepointDEBUG DataSourceTransactionManager:877 - Initiating transaction rollbackDEBUG DataSourceTransactionManager:347 - Rolling back JDBC transaction on Connection [com.mysql.cj.jdbc.ConnectionImpl@8ef162]DEBUG DataSourceTransactionManager:392 - Releasing JDBC Connection [com.mysql.cj.jdbc.ConnectionImpl@8ef162] after transactionjava.lang.ArithmeticException: / by zero... ...

发现整个事务都已经回滚了,按照回滚点的逻辑,addArea()方法抛出异常,不是应该只回滚到addArea()前吗,也就是addGood()应该被提交,这是为什么呢?

如果我们将addArea()方法try catch起来,就能得到我们想要的结果,addGood()被提交,而addArea()回滚,这又是为什么呢?我们带着这几个问题来分析源码

addAreaAndGood()开启事务

addAreaAndGood()开启事务,最外层方法使用传播属性PROPAGATION_REQUIRED、PROPAGATION_REQUIRES_NEW、PROPAGATION_NESTED效果都一样,都是开启一个新的事务。

org.springframework.transaction.support.AbstractPlatfORMTransactionManager#getTransaction

else if (def.getPropagationBehavior() == TransactionDefinition.PROPAGATION_REQUIRED ||def.getPropagationBehavior() == TransactionDefinition.PROPAGATION_REQUIRES_NEW ||def.getPropagationBehavior() == TransactionDefinition.PROPAGATION_NESTED) {// 第一次进来SuspendedResourcesHolder suspendedResources = suspend(null);if (debugEnabled) {logger.debug("Creating new transaction with name [" + def.getName() + "]: " + def);}try {// 开启新事务return startTransaction(def, transaction, debugEnabled, suspendedResources);}catch (RuntimeException | Error ex) {resume(null, suspendedResources);throw ex;}}

addGood()获得事务并创建回滚点

addGood()从ThreadLocal中获得addAreaAndGood()创建的事务,然后发现自己的传播属性为PROPAGATION_NESTED,就创建了一个回滚点。

org.springframework.transaction.support.AbstractPlatformTransactionManager#handleExistingTransaction

if (definition.getPropagationBehavior() == TransactionDefinition.PROPAGATION_NESTED) {if (!isNestedTransactionAllowed()) {throw new NestedTransactionNotSupportedException("Transaction manager does not allow nested transactions by default - " +"specify 'nestedTransactionAllowed' property with value 'true'");}if (debugEnabled) {logger.debug("Creating nested transaction with name [" + definition.getName() + "]");}if (useSavepointForNestedTransaction()) {// Create savepoint within existing Spring-managed transaction,// through the SavepointManager api implemented by TransactionStatus.// Usually uses JDBC 3.0 savepoints. Never activates Spring synchronization.DefaultTransactionStatus status =prepareTransactionStatus(definition, transaction, false, false, debugEnabled, null);// 创建回滚点status.createAndHoldSavepoint();return status;}else {// Nested transaction through nested begin and commit/rollback calls.// Usually only for JTA: Spring synchronization might get activated here// in case of a pre-existing JTA transaction.return startTransaction(definition, transaction, debugEnabled, null);}}

addGood()提交事务时释放回滚点

addGood()并不会真正的提交事务,因为事务并不是addGood()创建的,只是在提交时会将之前创建的回滚点释放。

org.springframework.transaction.support.AbstractPlatformTransactionManager#processCommit

if (status.hasSavepoint()) {// NESTED的提交if (status.isDebug()) {logger.debug("Releasing transaction savepoint");}unexpectedRollback = status.isGlobalRollbackOnly();// 只是释放回滚点status.releaseHeldSavepoint();}

addArea()获得事务并创建回滚点

流程与addGood()一致。

addArea()回滚事务释放回滚点

addArea()发生异常,会执行回滚事务的逻辑,并没有真正的回滚事务,因为事务并不是addArea()创建的,,只是将之前创建的回滚点释放。 org.springframework.transaction.support.AbstractPlatformTransactionManager#processRollback

if (status.hasSavepoint()) {// 用于NESTED传播机制,发生异常// 回滚至回滚点if (status.isDebug()) {logger.debug("Rolling back transaction to savepoint");}status.rollbackToHeldSavepoint();}

addAreaAndGood()回滚这个事务

addArea()发生异常后继续往外抛,addAreaAndGood()也会捕获到异常,然后执行回滚逻辑,这样整个事务都回滚了。 org.springframework.transaction.support.AbstractPlatformTransactionManager#processRollback

else if (status.isNewTransaction()) {// 只有最外层的事务newTransaction=trueif (status.isDebug()) {logger.debug("Initiating transaction rollback");}// 事务的回滚doRollback(status);}

为什么将addArea()方法try catch起来,整个事务就不会回滚了呢?

因为将addArea()方法try catch起来后,addAreaAndGood()就会执行提交事务的逻辑,这样addGood()就被提交了。

以上就是spring的事务传播属性REQUIRED_NESTED的原理介绍的详细内容了,看完之后是否有所收获呢?如果想了解更多相关内容,欢迎来编程网精选!

--结束END--

本文标题: spring的事务传播属性REQUIRED_NESTED的原理介绍

本文链接: https://www.lsjlt.com/news/358536.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • spring的事务传播属性REQUIRED_NESTED的原理介绍
    这篇文章主要介绍了spring的事务传播属性REQUIRED_NESTED的原理介绍,具有一定借鉴价值,需要的朋友可以参考下。下面就和我一起来看看吧。传统事务中回滚点的使用package com.morris.spring.dem...
    99+
    2023-07-06
  • spring的事务传播属性REQUIRED_NESTED原理
    目录传统事务中回滚点的使用事务的传播属性NESTEDaddAreaAndGood()开启事务addGood()获得事务并创建回滚点addGood()提交事务时释放回滚点addArea...
    99+
    2023-05-20
    spring 事务传播 事务传播属性 REQUIRED_NESTED 原理
  • spring的事务传播属性有哪些
    在Spring中,事务传播属性定义了一个方法的事务边界如何与其他方法的事务边界相互关联。Spring提供了以下的事务传播属性:1. ...
    99+
    2023-09-13
    spring
  • Spring中事务的传播属性总结
    本篇内容主要讲解“Spring中事务的传播属性总结”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“Spring中事务的传播属性总结”吧!我们用Spring时,很多时候都会用到他的声明式事务,简单的...
    99+
    2023-06-03
  • Spring事务的传播特性有哪些
    本文小编为大家详细介绍“Spring事务的传播特性有哪些”,内容详细,步骤清晰,细节处理妥当,希望这篇“Spring事务的传播特性有哪些”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识...
    99+
    2022-10-19
  • Spring中事务传播特性的示例分析
    这篇文章主要介绍Spring中事务传播特性的示例分析,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!一、使用方式可以采用Transactional,配置propagation即可。打开org.springframewo...
    99+
    2023-06-15
  • Spring是如何给事务实现传播特性的
    这篇文章主要介绍“Spring是如何给事务实现传播特性的”,在日常操作中,相信很多人在Spring是如何给事务实现传播特性的问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Spring是如何给事务实现传播特性的...
    99+
    2023-06-16
  • 关于spring事务传播行为非事务方式的理解
    目录spring事务传播行为非事务方式Spring事务理解和配置1 Spring事务1.1 事务简介1.2 事务的传播行为1.3 编程式事务(需要手动写代码,了解)1.4 声明式事务...
    99+
    2022-11-12
  • spring事务隔离级别、传播行为以及spring+mybatis+atomikos实现分布式事务管理的方法
    小编给大家分享一下spring事务隔离级别、传播行为以及spring+mybatis+atomikos实现分布式事务管理的方法,希望大家阅读完这篇文章后大所收获,下面让我们一起去探讨吧!1.事务的定义:事务...
    99+
    2022-10-18
  • Spring七大事务传递机制的实现原理是什么
    这篇“Spring七大事务传递机制的实现原理是什么”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“Spring七大事务传递机制...
    99+
    2023-07-05
  • Android中的广播、服务、数据库、通知、包等术语的原理和介绍(图解)
     简介  在Android中运用了很多机制,例如:广播、服务、数据库、通知、包……等等。 什么是广播?Broadcast是一种广泛运用的在应用程序之间传输信...
    99+
    2022-06-06
    数据 数据库 Android
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作