Spring的嵌套事务(Propagation.NESTED)到底是个啥

什么是嵌套事务

嵌套事务其实是对数据库SavePoint概念的Java操作版封装,什么是SavePoint参考我另一篇blog:juejin.cn/post/718544…

SavePoint是数据库事务中的一个概念, 可以将整个事务切割为不同的小事务, 可以选择将状态回滚到某个小事务发生时的样子。

Propagation.NESTED 则是外部事务的子事务, 如果外部事务 commit, 嵌套事务也会被 commit, 这个规则同样适用于rollback.

嵌套事务开始执行时,  它将取得一个 savepoint, 如果这个嵌套事务失败, 我们将回滚到此 savepoint,嵌套事务是外部事务的一部分, 只有外部事务结束后它才会被提交.

案例

伪代码如下:

ServiceA {  
      
    /** 
     * 事务属性配置为 PROPAGATION_REQUIRED 
     */  
    void methodA() {  
        try {
          DML......
          ServiceB.methodB();  
        } catch(...) ...
        DML....
    }  
  
}  
  
ServiceB {  
      
    /** 
     * 事务属性配置为 PROPAGATION_NESTED 
     */   
    void methodB() {  
    }  
     
}     

如上代码,当methodB开始执行时,spring会在数据库中生成一个SavePoint,methodB失败被spring捕捉到时,会回滚该SavePoint,将methodB中对数据库的操作全部回滚到SavePoint之前,

注意methodA中加入了try catch,引入你的代码如果希望methodB作为一个局部性的事务抛出异常失败后,不影响A的外围事务,那么就不能让异常继续往上抛,否则A这个外围事务也会被全部回滚,因此要在A中自行消化这个异常,

说到这里其实可以总结一下,嵌套事务就是spring像对正常事务一样,帮助我们捕捉SavePoint小事务的异常并进行自动回滚,免去我们在Java代码中使用SavePoint时想要回滚的手动操作,只是前者是一个正常的大事务回滚,后者是大事务中的小事务回滚。

下面是实际测试代码,我们在保存key的时候,也保存一个value,但是保存value时我们抛出exception,并且在保存key的地方catch住:

image.png

image.png

这里是保存前的db,都是空表:

image.png

image.png

操作后,可以看到key表保存进去,value表没保存进去:

image.png

image.png

这里有人会说,key表保存进去是因为我们自己吃了异常,没有被spring捕获到,事务提交了,所以key表保存进去了,value表只是因为被spring捕获到了而已,是不是和嵌套事务没关系?那我们改一下value的事务行为,也使用默认方式,代码如下,并且我们把上面保存进db的数据清空,方便查看效果:

image.png

执行后查看数据库, 空空如也:

image.png

image.png

我们再看log,spring出了一个 Transaction rolled back because it has been marked as rollback-only 的错误,这是什么原因呢? 更改了事务行为后就出现这种,

其实原因很简单,因为两个service都是用默认行为,那么他们同属一个事务,spring检测到了value的方法发生了错误后,会将事务标记为 rollback-only,意味着本次事务发生了错误必须回滚,后续如果尝试提交此事务,就会提示下面的错误信息,

但是我们上面使用嵌套事务时,并没有发生这种情况,因为保存value时,创建了一个SavePoint,spring捕捉到错误后回滚这个SavePoint,并不会将事务标记为错误回滚,后续key地方catch了这个错误,是可以正常提交此事务的。

image.png

SavePoint和嵌套事务在实际应用中场景不多,了解一下如果将来真有需要的话 可以派上用场。

这里有篇文章讲解嵌套事务,推荐一下:www.iteye.com/topic/35907

  • 5
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 2
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值