spring 事务传播机制

spring 事务 传播机制 描述的 事务方法直接相互调用,父子事物开启,挂起,回滚 等的处理方式。 绿色的 那几个 我认为比较重要。

1 , @Transactional(propagation=Propagation.REQUIRED)  默认值 等于 @Transactional

   有父方法传递过来的 事务环境,就在使用父方法的事务环境,如果父方法没有事务环境,那么就用新开启一个事物。

   备注:非事务执行的意思是单条sql就提交。

  例子设置: A 方法调用 B,C两个子方法。C执行后在C的方法中抛出异常

    情景1  条件:A 什么都不加,B 什么都不加,C,Propagation.REQUIRED ,

        结果:C 执行以后抛出异常。结果 B能写入 ,C写入的被回滚。

    解析:A,B都是非事物环境,C起了一个新的事务。 

    

    情景2  条件:A 加 @Transactional ,B 什么都不加,C,Propagation.REQUIRED ,

        结论:B C 的操作都回滚了

    解析:B,C都是用的 A开启的事务环境,C的时候抛出异常, 全部回滚。

  关键字:有就用,没有就新建一个

2 ,@Transactional( propagation=Propagation.SUPPORTS)

  如果父方法没有事物换机,那么当前方法就使用非事务环境执行,如果 有,就使用父方法的事务环境。 这个和只写一个@Transactional 几乎一样

  例子设置: A 方法调用 B,C两个子方法。C执行后在C的方法中抛出异常

    情景1  条件:A 什么都不加,B 什么都不加,C,Propagation.SUPPORTS,

        结果:B,C 都能写入成功

    解析:ABC,都是非事务环境 

    

   情景2  条件:A @Transactional ,B 什么都不加,C,Propagation.SUPPORTS,

        结果:B,C 都能写入后都回滚了,

    解析:ABC,都处于A开启的事务环境。

  关键字:有就用,没有就不用

3 @Transactional(propagation=Propagation.MANDATORY)

   当前方法必须处于事物环境才能执行。

 例子设置: A 方法调用 B,C两个子方法。

    情景1  条件:A 什么都不加,B 什么都不加,C,Propagation.MANDATORY,

        结果:执行C的时候抛出异常( No existing transaction found for transaction marked with propagation 'mandatory' )

    解析:C 需要事物环境才能执行

    

   情景2  条件:A @Transactional ,B 什么都不加,C,Propagation.MANDATORY,

        结果: 正常执行到 

    解析:C 需要事物环境才能执行

  关键字:必须要父类给我一个事务环境

4 @Transactional(propagation=Propagation.REQUIRES_NEW)

   必定要新开启一个事务。

 例子设置: A 方法调用 B,C两个子方法。C执行后在C的方法中抛出异常

    情景1  条件:A 什么都不加,B 什么都不加,C,Propagation.REQUIRES_NEW,

        结果:B 能写入,C 写入后被回滚了。

    解析:在父类没有事务环境的情况下,C启动了一个新的事务

    

   情景2  条件:A @Transactional ,B 什么都不加,C,Propagation.REQUIRES_NEW,

        结果: B,C都没有写入数据

    解析: 明显C 让父类事务一起回滚了。C如果 回滚,父类事物也会回滚。

  情景3  条件:A @Transactional ,B 什么都不加,C,Propagation.REQUIRES_NEW, 去掉 C执行后在C的方法中抛出异常,在A中 执行完C后抛出异常。

        结果: B的数据没写进去,C的数据写进去;额

    解析: B 使用的A开启的事物 ,C 开启了一个新事物。并且C执行完成以后就提交了,A 里面的异常让B一起回滚了。

  关键字:新建一个,父回滚子不回滚,子回滚,父也会滚。 并且子事物是在 子方法完成的时候提交的。

5 @Transactional(propagation=Propagation.NOT_SUPPORTED)

   当前方法一定不回用在事务环境执行,如果父类有事务,那么就先把它挂起。 和  REQUIRED 是相反的。

 例子设置: A 方法调用 B,C两个子方法。C执行后在C的方法中抛出异常

    情景1  条件:A @Transactional,B 什么都不加,C,Propagation.NOT_SUPPORTED,

        结果:B没有写入,C能写入。

    解析:B 写入了,然后被 C里面的 抛出的异常 给触发回退了,C 里面是非事执行,能够正常写入。

   关键字:不会在事务环境执行

6  @Transactional(propagation=Propagation.NEVER)

   和  MANDATORY 相反 ,这个方法只能执行在非事务环境,如果父类有事物,就抛出异常

 例子设置: A 方法调用 B,C两个子方法

    情景1  条件:@Transactional ,B 什么都不加,C,Propagation.NEVER,

        结果:抛出异常(Existing transaction found for transaction marked with propagation 'never')

    解析:NEVER 的父方法不能有 事物环境。

  关键字:父层不能有事务

7  @Transactional(propagation=Propagation.NESTED)

   内嵌事物,在没有外部事务 的时候 和 REQUIRED 一样, 这个和  Propagation.REQUIRES_NEW 容易混淆 ,Propagation.REQUIRES_NEW 是挂起父事物,然后新启动一个事物,新的事务完成后先提交,新事物回滚,父事物一起回滚,

    Propagation.NESTED 是在 父环境有事物的时候,会做一个保存点,然后在 如果 当前方法抛出异常,就回滚到保存点,如果 父事物 回滚,当前方法里面的写操作也回滚,而且本质上只有一个事物,所以在父方法事物提交的时候提交。

 例子设置: A 方法调用 B,C两个子方法,C方法内部执行完成以后抛出异常

    情景1  条件:@Transactional ,B 什么都不加,C,Propagation.NEVER,

        结果: B 没有写入成功,C也没有写入成功。

    解析:.这个结果和 REQUIRES_NEW 类似,不做解释,后面进一步确认。

     

    

   情景2  条件:@Transactional ,B 什么都不加,C,Propagation.NEVER, ,但是抛异常的位置改成A方法内部,C方法调用以后。

        结果: B 没有写入成功,C也没有写入成功。

    解析:综合和上面2个例子。NESTED 内层事物回滚,外层也会滚,外层回滚内层也会回滚。并且内层事物提交的 时间节点是和外层事务 一起提交。

   关键字:父子同步回滚,父子一起提交

 备注:sharding JDBC  不支持  @Transactional(propagation=Propagation.NESTED)

备注2:如果方法前面不写   @Transactional ,如果这个方法被有事务环境的 方法调用。 那么这个方法会使用 父类的 事务环境。如果父类没有就每条单独提交。

 

原文地址:https://www.cnblogs.com/cxygg/p/10710572.html