首页 > 其他分享 >Spring事务传播机制

Spring事务传播机制

时间:2024-01-16 17:00:43浏览次数:22  
标签:事务 -- Spring Transactional UPDATE 传播 方法

1.Spring对事物的支持一般有两种方式

  • 编程式事务管理:通过 TransactionTemplate或者TransactionManager手动管理事务,实际应用中很少使用,这不是本文的重点,就不在这里赘述。

  • 声明式事务管理:使用场景最多,也是最推荐使用的方式,直接加上@Transactional注解即可。

2.Transactional注解几大参数解释

@Transactional 注解是用于声明事务性方法的注解,通常用于标记在服务层的方法上。该注解提供了一些参数,用于配置事务的一些属性。以下是几个常用的参数及其解释:

  • propagation(传播行为):
  • - 用于指定事务的传播行为。包括诸如 `REQUIRED`、`REQUIRES_NEW`、`SUPPORTS`、`NOT_SUPPORTED` 等。该参数定义了方法被嵌套调用时,事务如何传播。
  • 2. isolation(隔离级别):
  • - 用于指定事务的隔离级别。包括 `DEFAULT`、`READ_UNCOMMITTED`、`READ_COMMITTED`、`REPEATABLE_READ`、`SERIALIZABLE`。该参数定义了事务处理过程中对数据的隔离程度。
  • 3. readOnly(只读):
  • - 用于指定事务是否是只读的。如果设置为 `true`,表示只读取数据而不修改,可以优化事务处理。默认值为 `false`。
  • 4. timeout`(超时时间):
  • - 用于指定事务的超时时间,单位为秒。如果事务执行时间超过指定的时间,则会被强制回滚。默认值为 `-1`,表示没有超时限制。
  • 5. rollbackFor` 和 noRollbackFor:
  • - 用于指定在哪些异常情况下回滚事务。`rollbackFor` 指定哪些异常时回滚,`noRollbackFor` 指定哪些异常时不回滚。可以传入异常类型的数组

 重点讲解propagation(传播行为)

七大参数设置: 

 场景:

假设有这样的场景 有A类和B类 A类内部有一个事务方法 B类有一个事务方法

class A{
    public B b;
    @Transactional
    public void a(){
        //1.对数据表A进行插入操作方法
        System.out.println("往A插入数据");
        //2.对数据表B进行插入操作方法
        b.b();
        //
        System.out.println("往A继续插入数据");
    }
}

class B{
    @Transactional
    public void b(){
        //对数据库进行插入操作方法
        System.out.println("更新表单 插入数据");
    }
}

这种是一种常见的嵌套事务 如果都存在事务可能它的SQL是这样的

BEGIN
 UPDATE A;
    -- B类的事务来了
    BEGIN
    UPDATE B;
    COMMIT;
    --     
UPDATE A;
COMMIT;

这种写法在MYSQL是不支持的,如果执行了B事务那么A的部门事务失效。如果想要实现这两个事务都存在怎么办?那我们可以直接把B事务的BEGIN和COMMIT去掉 让B事务融入到A事务中即可。

BEGIN
 UPDATE A;
    -- B类的事务来了
    UPDATE B;
    --     
UPDATE A;
COMMIT;

这种情况其实就代表了传播行为的 REQUIRED 传播行为

具体来说,当一个方法使用 @Transactional(propagation = Propagation.REQUIRED) 进行标记时,它的行为如下:

  1. 存在事务时: 方法将在当前事务中运行,与调用该方法的外部事务合并为一个事务。

  2. 不存在事务时: 方法将启动一个新的事务。

这种情况会出现一个问题 就是外部的事务融入到当前事务中的时候如果出错那么整个事务都会进行回滚。

解决这种情况有一个名词叫挂起 挂起解释就是使用其他线程获取不同的数据库连接 如果执行两个不同的事务 这样就不会影响原本事务的流程。

-- 线程1获取到数据库连接1 执行A事务的流程
BEGIN
 UPDATE A;
    -- B类的事务来了 线程2获取到数据库连接2 执行B事务的流程
    BEGIN
    UPDATE B;
    COMMIT;
    --  执行后 重新将线程B切换到线程A 来执行后续流程
UPDATE A;
COMMIT;

这种情况也是传播行为中的REQUIRES_NEW 传播行为

具体来说,当一个方法使用 @Transactional(propagation = Propagation.REQUIRES_NEW) 进行标记时,它的行为如下:

  1. 存在事务时: 方法将挂起当前的事务,并启动一个新的事务。在该方法执行完毕后,新事务将被提交或回滚,然后恢复挂起的事务。

  2. 不存在事务时: 方法将启动一个新的事务。

还有一种情况就是嵌套事务,MySQL是不支持嵌套事务的,但Mybatis在这个层面加入了保存点和回滚点来支持。

BEGIN
    UPDATE a set score=100 where id=1;
    Savepoint a;
    update b set score=200 where id=2;
    ROLLBACK to a;
--     如果B事务出现问题 不会影响后面的事务
        UPDATE a set score=300 where id=3;
COMMIT;

当一个方法使用 @Transactional(propagation = Propagation.NESTED) 进行标记时,它的行为如下:

  1. 存在事务时: 方法将在一个嵌套事务中运行。如果当前存在事务,则将创建一个新的保存点,并在方法执行时将该保存点设为当前事务的回滚点。如果方法执行后,嵌套事务回滚,则只会回滚到该保存点,而不会影响外部事务。

  2. 不存在事务时: 方法将启动一个新的事务,行为与 REQUIRED 类似。

上面三种情况其实就能解决掉百分之99%的事务嵌套问题,Spring为我们提供了七个事务传播行为,解释后四个

  • SUPPORTS:支持当前事务,B事务设置了这个传播行为,如果A有事务就融入到A事务中,如果没有就不开启事务,这个使用场景适用于只读和只有多SELECT场景下。
  •  MANDATORY:只支持当前事务,B事务如果设置了这个传播行为,那么它必须要被传入到一个有事务的方法中,不然就会抛出异常。
  • NOT_SUPPORTED:不支持事务,如果B事务设置了这个传播行为,那么它如果被传入到A事务方法内,那么它会将A事务进行挂起已非事务的方法来运行。
  • NEVER:不支持事务,只要有事务运行,就直接抛出异常。

 

标签:事务,--,Spring,Transactional,UPDATE,传播,方法
From: https://www.cnblogs.com/yunjie0930/p/17967919

相关文章

  • springboot~shardingsphere在非spring框架中的使用
    shardingsphere已经很方便的被springboot集成了,你只要引入sharding-jdbc-spring-boot-starter这个包就可以了,而如果是原生java的话,你就需要自己去实现了,主要是重新定义数据源,定义规则等问题,本文主要介绍原生环境下的shardingsphere的使用。依赖引用<dependencies><!--......
  • [spring] spring学习笔记(2): 通过xml实现依赖注入 - 特殊注入类型
    实际应用中,我们的对象可能会引用很多不同类型的东西,不单单只是几个数值对象类型在前一篇文章中,已经使用引用对象作为例子,关键在于使用ref<!--注意引用的对象要先创建Bean,id为weapon1--><beanid="player1"class="com.demo.player"> <!--通过setter注入,注意ref的......
  • springboot第48集:【思维导图】地图,面向对象,异常,功能代码
    在SpringBoot中,可以通过编写拦截器(Interceptor)来对请求进行拦截与处理。下面是一个简单的拦截器实现示例:创建一个类并实现HandlerInterceptor接口publicclassAuthInterceptorimplementsHandlerInterceptor{@OverridepublicbooleanpreHandle(HttpServletRequest......
  • Springboot上传文件大小限制处理
    今天在开发过程中遇到一个文件上传的问题io.undertow.server.RequestTooBigException:UT000020:Connectionterminatedasrequestwaslargerthan10485760Servlet容器使用的是undertow,看异常信息应该是默认存在10MB的文件大小限制。百度了一下,找到如下配置,问题得以解决,记......
  • 无涯教程-SQL - Transactions(事务)
    事务是将一个或多个更改打包在一起保存到数据库,事务对于确保数据完整性和处理数据库错误很重要。事务性质事务具有以下四个标准属性,通常以首字母缩写ACID表示。原子性-确保工作单元内的所有操作均成功完成,否则,事务将在失败时中止,并且所有先前的操作都将还原到以前的状态......
  • Spring Cloud整体架构解析
    SpringCloud整体架构SpringCloud的中文名我们就暂且称呼它为“春云”吧,听上去是多么朴实无华的名字,不过呢一般名字起的低调的都是厉害角色,我们就看看SpringCloud都提供了哪些靠谱功能吧。SpringCloud是一款微服务架构的一站式解决方案,你在微服务化过程中碰到的任何问题,都可......
  • SpringSecurity表单认证(二)
    用户名+密码系统默认登录用户名:user密码每次服务启动后随机生成密码用户信息获取原理(数据库获取)实现该接口,security默认自动生成密码关闭。框架源码:packageorg.springframework.security.core.userdetails;publicinterfaceUserDetailsService{UserDetailsloa......
  • [spring] spring学习笔记(1): 通过xml实现依赖注入(1)
    依赖注入是spring框架中一个重要思想-InversionofControl(IoC)-的实现,大体上来说,就是通过配置Bean对象,让spring内置的方法来为主对象创建需要的依赖对象;打个比方,在java中,当我们想要使用某个类时,应当通过new关键字来指定,i.e.//在这里创建一个类,他需要使......
  • Spring 工具:DigestUtils md5 摘要工具
    工具类:org.springframework.util.DigestUtils作用:计算字节数组、输入流的md5摘要所在模块:spring-core方法描述Stringmd5DigestAsHex(byte[]bytes)返回字节数组的md5摘要(计算字符串)Stringmd5DigestAsHex(InputStreaminputStream)返回输入流的md5......
  • Spring 工具:StopWatch 计时器
    工具类:org.springframework.util.StopWatch作用:记录方法执行耗时,统计每个方法的耗时占比所在模块:spring-core方法描述voidstart(StringtaskName)开始一个新的监测任务,可设置任务名称。记录当前时间和任务名称voidstop()结束当前监测任务。记录任务执行......