【Spring】Spring的7种事务传播机制讲解,以及49种组合方式介绍

2024-06-04 1962阅读

文章目录

目录

  • 文章目录
  • 正文
  •         一、Spring的7种事务传播机制
  •         二、事务传播的49种组合方式

    正文

            一、Spring的7种事务传播机制

            Spring事务传播机制是指在多个事务方法相互调用的情况下,事务如何在方法之间进行传播和管理的机制。通过事务传播机制,可以控制事务的范围和行为,保证多个事务方法的一致性和完整性。Spring提供了七种事务传播行为,分别是:REQUIRED、SUPPORTS、MANDATORY、REQUIRES_NEW、NOT_SUPPORTED、NEVER、NESTED。其中REQUIRED为Spring默认的传播机制。

            1. REQUIRED:如果当前存在事务,则加入该事务,如果不存在事务,则创建一个事务。

             解释:当方法a声明了该事务,方法b也声明该事务时,如果方法a调用了方法b,那么b不会再创建自己事务而是加入方法a,他们共享同一个事务上下文,b回滚a也回滚、a回滚b也回滚。而当方法a未声明事务,方法b声明了该事务时,a调用b,会为方法b创建事务,而a不会。

    @Transactional(propagation = Propagation.REQUIRED)  //默认@Transactional即可

             REQUIRED常用于有修改操作的操作。

            2. SUPPORTS:如果当前存在事务,则加入该事务,如果不存在事务,则以非事务的方式执行。

            解释:当方法a声明了一个事务,方法b声明了该事务时,同理,他们共享事务上下文。而当方法a未声明事务,方法b声明该事务,a调用b,a、b都没有事务,出现异常时a、b都不回滚。当然如果有一个方法c声明了该事务,我们直接调用该方法,出现异常也不会回滚,因为主方法是未声明事务的因此c也以非事务的方式执行。

    @Transactional(propagation = Propagation.SUPPORTS)

             SUPPORTS常用于只有查询相关的操作:当只需要查询时不开启事务,而当需要根据这些查询结果再进行修改操作时,加入到修改操作方法的事务中去,从而保证原子性。

            3. MANDATORY:如果当前存在事务,则加入到当前事务中;如果当前没有事务,则抛出异常。

            解释:当方法a声明了一个事务,方法b声明了该事务时,同理,他们共享事务上下文。而当方法a未声明事务,方法b声明该事务,a调用b,抛出异常。同理如果有一个方法c声明了该事务,我们直接调用该方法,由于主方法未声明事务因此会直接抛出异常。

    @Transactional(propagation = Propagation.MANDATORY)

            4. REQUIRES_NEW:总是创建一个新的事务,如果当前存在事务,则挂起当前事务。

            解释:不论方法a是否声明事务,方法b声明了该事务时,a调用b,都会为b创建一个自己的事务。而当a没有事务时,b独立事务运行;当a有事务时,挂起a的事务,等待b的事务执行完毕后再恢复执行a。

    @Transactional(propagation = Propagation.REQUIRES_NEW)

            5. NOT_SUPPORTED:以非事务方式执行操作,如果当前存在事务,则挂起当前事务。

            解释:如果方法a存在事务,会挂起a的事务,等待b执行完毕后再恢复执行a。

    @Transactional(propagation = Propagation.NOT_SUPPORTED)

            6. NEVER:以非事务方式执行操作,如果当前存在事务,则抛出异常。

            解释:当方法a声明了一个事务,方法b声明了该事务时,a调用b,b方法会抛出异常。

    @Transactional(propagation = Propagation.NEVER)

            7. NESTED:如果当前存在事务,则在当前事务中创建一个新的嵌套事务;如果当前没有事务,则创建一个新的事务。

             解释:如果数据库支持嵌套事务,那么就会存在多个begin和commit,a调用b,方法b运行完就提交了,与a互不影响。如果数据库不支持嵌套事务比如常用的MySQL,那么使用savepoint设置保存点,如果代码出现问题需要回滚,就使用rollback返回到保持点,最后在代码执行完后commit。

    @Transactional(propagation = Propagation.NESTED)

            二、事务传播的49种组合方式

            这里都以方法a和方法b举例,a调用b。

    1. 方法a事务传播方式为REQUIRED时

    REQUIRED - REQUIRED:如果a方法在事务中运行,则b方法将继承a方法的事务。如果a方法没有事务,则b方法将在一个新的事务中运行。
    REQUIRED - SUPPORTS:如果a方法在事务中运行,则b方法将继承a方法的事务。如果a方法没有事务,则b方法将以非事务方式运行。
    REQUIRED - MANDATORY:如果a方法在事务中运行,则b方法将继承a方法的事务。如果a方法没有事务,则b方法将抛出异常。
    REQUIRED - REQUIRES_NEW:无论a方法是否在事务中运行,b方法都会开启一个新的事务并在该事务中运行。
    REQUIRED - NOT_SUPPORTED:如果a方法在事务中运行,则b方法将以非事务方式运行。如果a方法没有事务,则b方法也将以非事务方式运行。
    REQUIRED - NEVER:如果a方法在事务中运行,则b方法将抛出异常。如果a方法没有事务,则b方法将以非事务方式运行。
    REQUIRED - NESTED:如果a方法在事务中运行,则b方法将作为a方法的嵌套事务运行。如果a方法没有事务,则b方法将在一个新的事务中运行。

    2. 方法a事务传播方式为SUPPORTS时

    SUPPORTS - REQUIRED:如果a方法在事务中运行,则b方法将继承a方法的事务。如果a方法没有事务,则b方法将在一个新的事务中运行。
    SUPPORTS - SUPPORTS:无论a方法是否在事务中运行,b方法都不会开启新的事务,而是以非事务方式运行。
    SUPPORTS - MANDATORY:如果a方法在事务中运行,则b方法将继承a方法的事务。如果a方法没有事务,则b方法将抛出异常。
    SUPPORTS - REQUIRES_NEW:无论a方法是否在事务中运行,b方法都会开启一个新的事务并在该事务中运行。
    SUPPORTS - NOT_SUPPORTED:无论a方法是否在事务中运行,b方法都将以非事务方式运行。
    SUPPORTS - NEVER:如果a方法在事务中运行,则b方法将抛出异常。如果a方法没有事务,则b方法将以非事务方式运行。
    SUPPORTS - NESTED:如果a方法在事务中运行,则b方法将作为a方法的嵌套事务运行。如果a方法没有事务,则b方法将以非事务方式运行。

     3. 方法a事务传播方式为MANDATORY时

    MANDATORY - REQUIRED:如果a方法在事务中运行,则b方法将继承a方法的事务。如果a方法没有事务,则b方法将抛出异常。
    MANDATORY - SUPPORTS:如果a方法在事务中运行,则b方法将继承a方法的事务。如果a方法没有事务,则b方法将以非事务方式运行。
    MANDATORY - MANDATORY:如果a方法在事务中运行,则b方法将继承a方法的事务。如果a方法没有事务,则b方法将抛出异常。
    MANDATORY - REQUIRES_NEW:无论a方法是否在事务中运行,b方法都会开启一个新的事务并在该事务中运行。
    MANDATORY - NOT_SUPPORTED:如果a方法在事务中运行,则b方法将以非事务方式运行。如果a方法没有事务,则b方法也将以非事务方式运行。
    MANDATORY - NEVER:如果a方法在事务中运行,则b方法将抛出异常。如果a方法没有事务,则b方法将以非事务方式运行。
    MANDATORY - NESTED:如果a方法在事务中运行,则b方法将作为a方法的嵌套事务运行。如果a方法没有事务,则b方法将抛出异常。

    4. 方法a事务传播方式为REQUIRES_NEW时

    REQUIRES_NEW - REQUIRED:无论a方法是否在事务中运行,b方法都会开启一个新的事务并在该事务中运行。
    REQUIRES_NEW - SUPPORTS:无论a方法是否在事务中运行,b方法都会开启一个新的事务并在该事务中运行。
    REQUIRES_NEW - MANDATORY:无论a方法是否在事务中运行,b方法都会开启一个新的事务并在该事务中运行。
    REQUIRES_NEW - REQUIRES_NEW:无论a方法是否在事务中运行,b方法都会开启一个新的事务并在该事务中运行。
    REQUIRES_NEW - NOT_SUPPORTED:无论a方法是否在事务中运行,b方法都将以非事务方式运行。
    REQUIRES_NEW - NEVER:无论a方法是否在事务中运行,b方法都会以非事务方式运行。
    REQUIRES_NEW - NESTED:无论a方法是否在事务中运行,b方法都会开启一个新的事务并在该事务中运行。

    5. 方法a事务传播方式为NOT_SUPPORTED时

    NOT_SUPPORTED - REQUIRED:如果a方法在事务中运行,则b方法将以非事务方式运行。如果a方法没有事务,则b方法将在一个新的事务中运行。
    NOT_SUPPORTED - SUPPORTS:无论a方法是否在事务中运行,b方法都将以非事务方式运行。
    NOT_SUPPORTED - MANDATORY:如果a方法在事务中运行,则b方法将以非事务方式运行。如果a方法没有事务,则b方法将抛出异常。
    NOT_SUPPORTED - REQUIRES_NEW:无论a方法是否在事务中运行,b方法都会开启一个新的事务并在该事务中运行。
    NOT_SUPPORTED - NOT_SUPPORTED:无论a方法是否在事务中运行,b方法都将以非事务方式运行。
    NOT_SUPPORTED - NEVER:如果a方法在事务中运行,则b方法将抛出异常。如果a方法没有事务,则b方法将以非事务方式运行。
    NOT_SUPPORTED - NESTED:如果a方法在事务中运行,则b方法将以非事务方式运行。如果a方法没有事务,则b方法将在一个新的事务中运行。

    6. 方法a事务传播方式为NEVER时

    NEVER - REQUIRED:如果a方法在事务中运行,则b方法将抛出异常。如果a方法没有事务,则b方法将在一个新的事务中运行。
    NEVER - SUPPORTS:如果a方法在事务中运行,则b方法将抛出异常。如果a方法没有事务,则b方法将以非事务方式运行。
    NEVER - MANDATORY:如果a方法在事务中运行,则b方法将抛出异常。如果a方法没有事务,则b方法将抛出异常。
    NEVER - REQUIRES_NEW:无论a方法是否在事务中运行,b方法都会开启一个新的事务并在该事务中运行。
    NEVER - NOT_SUPPORTED:无论a方法是否在事务中运行,b方法都将以非事务方式运行。
    NEVER - NEVER:如果a方法在事务中运行,则b方法将抛出异常。如果a方法没有事务,则b方法将以非事务方式运行。
    NEVER - NESTED:如果a方法在事务中运行,则b方法将抛出异常。如果a方法没有事务,则b方法将以非事务方式运行。

    7. 方法a事务传播方式为NESTED时

    NESTED - REQUIRED:如果a方法在事务中运行,则b方法将作为a方法的嵌套事务运行。如果a方法没有事务,则b方法将在一个新的事务中运行。
    NESTED - SUPPORTS:如果a方法在事务中运行,则b方法将作为a方法的嵌套事务运行。如果a方法没有事务,则b方法将以非事务方式运行。
    NESTED - MANDATORY:如果a方法在事务中运行,则b方法将作为a方法的嵌套事务运行。如果a方法没有事务,则b方法将抛出异常。
    NESTED - REQUIRES_NEW:无论a方法是否在事务中运行,b方法都会开启一个新的事务并在该事务中运行。
    NESTED - NOT_SUPPORTED:如果a方法在事务中运行,则b方法将以非事务方式运行。如果a方法没有事务,则b方法将以非事务方式运行。
    NESTED - NEVER:如果a方法在事务中运行,则b方法将抛出异常。如果a方法没有事务,则b方法将以非事务方式运行。
    NESTED - NESTED:如果a方法在事务中运行,则b方法将作为a方法的嵌套事务运行。如果a方法没有事务,则b方法将在一个新的事务中运行。

    免责声明:我们致力于保护作者版权,注重分享,被刊用文章因无法核实真实出处,未能及时与作者取得联系,或有版权异议的,请联系管理员,我们会立即处理! 部分文章是来自自研大数据AI进行生成,内容摘自(百度百科,百度知道,头条百科,中国民法典,刑法,牛津词典,新华词典,汉语词典,国家院校,科普平台)等数据,内容仅供学习参考,不准确地方联系删除处理! 图片声明:本站部分配图来自人工智能系统AI生成,觅知网授权图片,PxHere摄影无版权图库和百度,360,搜狗等多加搜索引擎自动关键词搜索配图,如有侵权的图片,请第一时间联系我们,邮箱:ciyunidc@ciyunshuju.com。本站只作为美观性配图使用,无任何非法侵犯第三方意图,一切解释权归图片著作权方,本站不承担任何责任。如有恶意碰瓷者,必当奉陪到底严惩不贷!

    目录[+]