阿里大牛带你深刻分析spring事务传播行为

spring框架封装了不少有用的功能和组件,便于在项目开发中快速高效的调用,其中spring的事务使用很是简单,只须要在用到事务的地方加一行注解便可:spring

1@Transactionalsql

但越是看起来简单的东西,就越多坑,为什么如此?数据库

不是由于别人造的轮子有问题,而是咱们在用轮子的时候,只会用,不知其原理,于是不能在实际使用场景中很好的把握它。架构

spring的事务传播行为在实战中是很是重要的,但90%的人只会@Transactional,在此颇有必要给你们分享一下。并发

在事务注解的参数中,咱们能够看到有这样的一个参数:框架

 

 

这个参数就是用来控制事务传播行为的,下面对这个参数进行详细的解答每个参数表明的传播行为以及在讲解在实际场景中应该选择参数值。分布式

文章提纲:高并发

Spring事务传播行为性能

事务传播行为剖析(实例)学习

参数行为场景应用(实例)

1.Spring事务传播行为:

spring支持7种事务传播行为,肯定客户端和被调用端的事务边界(说得通俗一点就是多个具备事务控制的service的相互调用时所造成的复杂的事务边界控制)下图所示为7钟事务传播机制:

传播行为(XML)含义

REQUIRED:表示当前方法必须在一个具备事务的上下文中运行,若有客户端有事务在进行,那么被调用端将在该事务中运行,不然的话从新开启一个事务(若是被调用端发生异常,那么调用端和被调用端事务都将回滚)。

SUPPORTS:表示当前方法没必要须要具备一个事务上下文,可是若是有一个事务的话,它也能够在这个事务中运行。

MANDATORY:表示当前方法必须在一个事务中运行,若是没有事务,将抛出异常

NESTED:表示若是当前方法正有一个事务在运行中,则该方法应该运行在一个嵌套事务中,被嵌套的事务能够独立于被封装的事务中进行提交或者回滚。若是封装事务存在,而且外层事务抛出异常回滚,那么内层事务必须回滚,反之,内层事务并不影响外层事务。若是封装事务不存在,则同REQUIRED的同样。

NEVER:表示当方法务不该该在一个事务中运行,若是存在一个事务,则抛出异常。

REQUIRES_NEW:表示当前方法必须运行在它本身的事务中。一个新的事务将启动,并且若是有一个现有的事务在运行的话,则这个方法将在运行期被挂起,直到新的事务提交或者回滚才恢复执行。

NOT_SUPPORTED:表示该方法不该该在一个事务中运行。若是有一个事务正在运行,他将在运行期被挂起,直到这个事务提交或者回滚才恢复执行。

2.事务传播行为剖析(实例)

假设有类A的方法methodB(),有类B的方法methodB().

REQUIRED

若是B的方法methodB()的事务传播特性是required,那么以下图

 

 

A.methodA()调用B的methodB()方法,那么若是A的方法包含事务,则B的方法则不重新开启事务,

一、 若是B的methodB()抛出异常,A的methodB()没有捕获,则A和B的事务都会回滚;

二、 若是B的methodB()运行期间异常会致使B的methodB()的回滚,A若是捕获了异常,并正常提交事务,则会发生Transaction rolled back because it has been marked as rollback-only的异常。

三、 若是A的methodA()运行期间异常,则A和B的Method的事务都会被回滚

SUPPORTS

若是B的方法methodB()的事务传播特性是supports,么以下图

 

A.methodA()调用B的methodB()方法,那么若是A的方法包含事务,则B运行在此事务环境中,若是A的方法不包含事务,则B运行在非事务环境;

一、若是A没有事务,则A和B的运行出现异常都不会回滚。

二、若是A有事务,A的method方法执行抛出异常,B.methodB和A.methodA都会回滚。

三、若是A有事务,B.method抛出异常,B.methodB和A.methodA都会回滚,若是A捕获了B.method抛出的异常,则会出现异常Transactionrolled back because it has been marked as rollback-only。

MANDATORY

表示当前方法必须在一个事务中运行,若是没有事务,将抛出异常,以下图调用关系:

 

 

B.methodB()事务传播特性定义为:MANDATORY

一、若是A的methoda()方法没有事务运行环境,则B的methodB()执行的时候会报以下异常:No existingtransaction found for transaction marked with propagation 'mandatory'

二、若是A的Methoda()方法有事务而且执行过程当中抛出异常,则A.methoda()和B.methodb()执行的操做被回滚;

三、若是A的methoda()方法有事务,则B.methodB()抛出异常时,A的methoda()和B.methodB()都会被回滚;若是A捕获了B.method抛出的异常,则会出现异常Transaction rolled back because ithas been marked as rollback-only

NESTED

若有一下方法调用关系,如图:

 

 

B的methodB()定义的事务为NESTED;

一、若是A的MethodA()不存在事务,则B的methodB()运行在一个新的事务中,B.method()抛出的异常,B.methodB()回滚,但A.methodA()不回滚;若是A.methoda()抛出异常,则A.methodA()和B.methodB()操做不回。

二、 若是A的methodA()存在事务,则A的methoda()抛出异常,则A的methoda()和B的Methodb()都会被回滚;

三、 若是A的MethodA()存在事务,则B的methodB()抛出异常,B.methodB()回滚,若是A不捕获异常,则A.methodA()和B.methodB()都会回滚,若是A捕获异常,则B.methodB()回滚,A不回滚;

NEVER

表示事务传播特性定义为NEVER的方法不该该运行在一个事务环境中

有以下调用关系:

 

 

若是B.methodB()的事务传播特性被定义为NEVER,则若是A.methodA()方法存在事务,则会出现异常Existingtransaction found for transaction marked with propagation 'never'。

REQUIRES_NEW

表示事务传播特性定义为REQUIRES_NEW的方法须要运行在一个新的事务中。

若有一下调用关系:B.methodB()事务传播特性为REQUIRES_NEW.

 

 

一、 若是A存在事务,A.methodA()抛出异常,A.methodA()的事务被回滚,但B.methodB()事务不受影响;若是B.methodB()抛出异常,A不捕获的话,A.methodA()和B.methodB()的事务都会被回滚。若是A捕获的话,A.methodA()的事务不受影响但B.methodB()的事务回滚。

NOT_SUPPORTED

表示该方法不该该在一个事务中运行。若是有一个事务正在运行,他将在运行期被挂起,直到这个事务提交或者回滚才恢复执行。

若有一下调用关系图:

 

 

若是B.methodB()方法传播特性被定义为:NOT_SUPPORTED。

一、 若是A.methodA()存在事务,若是B.methodB()抛出异常,A.methodA()不捕获的话,A.methodA()的事务被回滚,而B.methodB()出现异常前数据库操做不受影响。若是A.methodA()捕获的话,则A.methodA()的事务不受影响,B.methodB()异常抛出前的数据操做不受影响。

3. 参数行为场景应用(实例)

一、 在一个话费充值业务处理逻辑中,有以下图所示操做:

 

 

业务须要扣款操做和建立订单操做同成功或者失败,所以,charger()和order()的事务不能相互独立,须要包含在chargeHandle()的事务中;

经过以上需求,能够给charge()和order()的事务传播行为定义成:MANDATORY

只要charge()或者order()抛出异常整个chargeHandle()都一块儿回滚,即便chargeHandle()捕获异常也没用,不容许提交事务。

二、 若是业务需求没接受到一次请求到要记录日志到数据库,以下图:

 

 

由于log()的操做无论扣款和建立订单成功与否都要生成日志,而且日志的操做成功与否不影响充值处理,因此log()方法的事务传播行为能够定义为:REQUIRES_NEW.

在订单的售后处理中,更新完订单金额后,须要自动统计销售报表,以下图所示:

 

 

根据业务可知,售后是已经处理完订单的充值请求后的功能,是对订单的后续管理,统计报表report()方法耗时较长,所以,咱们须要设置report()的事务传播行为为:NEVER,表示不适合在有事务的操做中调用,由于report()太耗时。

在银行新增银行卡业务中,须要执行两个操做,一个是保存银行卡信息,一个是登记新建立的银行卡信息,其中登记银行卡信息成功与否不影响银行卡的建立。

 

由以上需求,咱们可知对于regster()方法的事务传播行为,能够设置为NESTED,action()事务的回滚,regster()保存的信息就没意义,也就须要跟着回滚,而regster()的回滚不影响action()事务;insert()的事务传播行为能够设置为REQUIRED, MANDATORY,即insert()回滚事务,action()的事务必须跟着回滚。

欢迎工做一到五年的Java工程师朋友们加入Java高级架构:705127209 

群内提供免费的Java架构学习资料(里面有高可用、高并发、高性能及分布式、Jvm性能调优、Spring源码,

MyBatis,Netty,Redis,Kafka,Mysql,Zookeeper,Tomcat,Docker,Dubbo,Nginx等多个知识点的架构资料)

合理利用本身每一分每一秒的时间来学习提高本身,不要再用"没有时间“来掩饰本身思想上的懒惰!趁年轻,使劲拼,给将来的本身一个交代!

相关文章
相关标签/搜索