前言

在之前的文章中已经对Spring中的事务做了详细的分析了,这篇文章我们来聊一聊平常工作时使用事务可能出现的一些问题(本文主要针对使用@Transactional进行事务管理的方式进行讨论)以及对应的解决方案

  1. 事务失效

  2. 事务回滚相关问题

  3. 读写分离跟事务结合使用时的问题

事务失效

事务失效我们一般要从两个方面排查问题

数据库层面

数据库层面,数据库使用的存储引擎是否支持事务?默认情况下MySQL数据库使用的是Innodb存储引擎(5.5版本之后),它是支持事务的,但是如果你的表特地修改了存储引擎,例如,你通过下面的语句修改了表使用的存储引擎为MyISAM,而MyISAM又是不支持事务的

alter table table_name engine=myisam;

这样就会出现“事务失效”的问题了

「解决方案」:修改存储引擎为Innodb

业务代码层面

业务层面的代码是否有问题,这就有很多种可能了

  1. 我们要使用Spring的声明式事务,那么需要执行事务的Bean是否已经交由了Spring管理?在代码中的体现就是类上是否有@ServiceComponent等一系列注解

「解决方案」:将Bean交由Spring进行管理(添加@Service注解)

  1. @Transactional注解是否被放在了合适的位置。在上篇文章中我们对Spring中事务失效的原理做了详细的分析,其中也分析了Spring内部是如何解析@Transactional注解的,我们稍微回顾下代码:

注解解析

❝代码位于:AbstractFallbackTransactionAttributeSource#computeTransactionAttribute中❞

也就是说,默认情况下你无法使用@Transactional对一个非public的方法进行事务管理

「解决方案」:修改需要事务管理的方法为public

  1. 出现了自调用。什么是自调用呢?我们看个例子

@Servicepublic class DmzService {public void saveAB(A a, B b) {
  saveA(a);
  saveB(b);
 }@Transactionalpublic void saveA(A a) {
  dao.saveA(a);
 }@Transactionalpublic void saveB(B b){
  dao.saveB(a);
 }
}

上面三个方法都在同一个类DmzService中,其中saveAB方法中调用了本类中的saveAsaveB方法,这就是自调用。在上面的例子中saveAsaveB上的事务会失效

那么自调用为什么会导致事务失效呢?我们知道Spring中事务的实现是依赖于AOP的,当容器在创建dmzService这个Bean时,发现这个类中存在了被@Transactional标注的方法(修饰符为public)那么就需要为这个类创建一个代理对象并放入到容器中,创建的代理对象等价于下面这个类

public class DmzServiceProxy {private DmzService dmzService;public DmzServiceProxy(DmzService dmzService) {this.dmzService = dmzService;
    }public void saveAB(A a, B b) {
        dmzService.saveAB(a, b);
    }public void saveA(A a) {try {// 开启事务
            startTransaction();
            dmzService.saveA(a);
        } catch (Exception e) {// 出现异常回滚事务
            rollbackTransaction();
        }// 提交事务
        commitTransaction();
    }public void saveB(B b) {try {// 开启事务
            startTransaction();
            dmzService.saveB(b);
        } catch (Exception e) {// 出现异常回滚事务
            rollbackTransaction();
        }// 提交事务
        commitTransaction();
    }
}

上面是一段伪代码,通过startTransactionrollbackTransactioncommitTransaction这三个方法模拟代理类实现的逻辑。因为目标类DmzService中的saveAsaveB方法上存在@Transactional注解,所以会对这两个方法进行拦截并嵌入事务管理的逻辑,同时saveAB方法上没有@Transactional,相当于代理类直接调用了目标类中的方法。

我们会发现当通过代理类调用saveAB时整个方法的调用链如下:

实际上我们在调用saveAsaveB时调用的是目标类中的方法,这种清空下,事务当然会失效。

常见的自调用导致的事务失效还有一个例子,如下:

@Servicepublic class DmzService {@Transactionalpublic void save(A a, B b) {
  saveB(b);
 }@Transactional(propagation = Propagation.REQUIRES_NEW)public void saveB(B b){
  dao.saveB(a);
 }
}

当我们调用save方法时,我们预期的执行流程是这样的

也就是说两个事务之间互不干扰,每个事务都有自己的开启、回滚、提交操作。

但根据之前的分析我们知道,实际上在调用saveB方法时,是直接调用的目标类中的saveB方法,在saveB方法前后并不会有事务的开启或者提交、回滚等操作,实际的流程是下面这样的

由于saveB方法实际上是由dmzService也就是目标类自己调用的,所以在saveB方法的前后并不会执行事务的相关操作。这也是自调用带来问题的根本原因:「自调用时,调用的是目标类中的方法而不是代理类中的方法」

「解决方案」

  1. 自己注入自己,然后显示的调用,例如:

    @Servicepublic class DmzService {// 自己注入自己@Autowired DmzService dmzService;@Transactionalpublic void save(A a, B b) {  dmzService.saveB(b); }@Transactional(propagation = Propagation.REQUIRES_NEW)public void saveB(B b){  dao.saveB(a); }}

    这种方案看起来不是很优雅利用AopContext,如下:

    @Servicepublic class DmzService {@Transactionalpublic void save(A a, B b) {  ((DmzService) AopContext.currentProxy()).saveB(b); }@Transactional(propagation = Propagation.REQUIRES_NEW)public void saveB(B b){  dao.saveB(a); }}

    ❝使用上面这种解决方案需要注意的是,需要在配置类上新增一个配置

    // exposeProxy=true代表将代理类放入到线程上下文中,默认是false@EnableAspectJAutoProxy(exposeProxy = true)

    ❞个人比较喜欢的是第二种方式

这里我们做个来做个小总结

总结

一图胜千言

事务失效的原因

事务回滚相关问题

回滚相关的问题可以被总结为两句话

  1. 想回滚的时候事务却提交了
  2. 想提交的时候被标记成只能回滚了(rollback only)

先看第一种情况:「想回滚的时候事务却提交了」。这种情况往往是程序员对Spring中事务的rollbackFor属性不够了解导致的。❝Spring默认抛出了未检查unchecked异常(继承自 RuntimeException的异常)或者 Error才回滚事务;其他异常不会触发回滚事务,已经执行的SQL会提交掉。如果在事务中抛出其他类型的异常,但却期望 Spring 能够回滚事务,就需要指定rollbackFor属性。❞对应代码其实我们上篇文章也分析过了,如下:

回滚代码

❝以上代码位于:TransactionAspectSupport#completeTransactionAfterThrowing方法中❞默认情况下,只有出现RuntimeException或者Error才会回滚

public boolean rollbackOn(Throwable ex) {return (ex instanceof RuntimeException || ex instanceof Error);}

所以,如果你想在出现了非RuntimeException或者Error时也回滚,请指定回滚时的异常,例如:

@Transactional(rollbackFor = Exception.class)

第二种情况:「想提交的时候被标记成只能回滚了(rollback only)」。对应的异常信息如下:

Transaction rolled back because it has been marked as rollback-only

我们先来看个例子吧

@Servicepublic class DmzService {@Autowired IndexService indexService;@Transactionalpublic void testRollbackOnly() {try {   indexService.a();  } catch (ClassNotFoundException e) {   System.out.println("catch");  } }}@Servicepublic class IndexService {@Transactional(rollbackFor = Exception.class)public void a() throws ClassNotFoundException{// ......throw new ClassNotFoundException(); }}

在上面这个例子中,DmzServicetestRollbackOnly方法跟IndexServicea方法都开启了事务,并且事务的传播级别为required,所以当我们在testRollbackOnly中调用IndexServicea方法时这两个方法应当是共用的一个事务。按照这种思路,虽然IndexServicea方法抛出了异常,但是我们在testRollbackOnly将异常捕获了,那么这个事务应该是可以正常提交的,为什么会抛出异常呢?如果你看过我之前的源码分析的文章应该知道,在处理回滚时有这么一段代码

rollBackOnly设置

在提交时又做了下面这个判断(这个方法我删掉了一些不重要的代码)

commit_rollbackOnly

可以看到当提交时发现事务已经被标记为rollbackOnly后会进入回滚处理中,并且unexpected传入的为true。在处理回滚时又有下面这段代码

抛出异常

最后在这里抛出了这个异常。❝以上代码均位于AbstractPlatformTransactionManager中❞总结起来,「主要的原因就是因为内部事务回滚时将整个大事务做了一个rollbackOnly的标记」,所以即使我们在外部事务中catch了抛出的异常,整个事务仍然无法正常提交,并且如果你希望正常提交,Spring还会抛出一个异常。「解决方案」:这个解决方案要依赖业务而定,你要明确你想要的结果是什么

  1. 内部事务发生异常,外部事务catch异常后,内部事务自行回滚,不影响外部事务

❝将内部事务的传播级别设置为nested/requires_new均可。在我们的例子中就是做如下修改:

// @Transactional(rollbackFor = Exception.class,propagation = Propagation.REQUIRES_NEW)@Transactional(rollbackFor = Exception.class,propagation = Propagation.NESTED)public void a() throws ClassNotFoundException{// ......throw new ClassNotFoundException();}

❞虽然这两者都能得到上面的结果,但是它们之间还是有不同的。当传播级别为requires_new时,两个事务完全没有联系,各自都有自己的事务管理机制(开启事务、关闭事务、回滚事务)。但是传播级别为nested时,实际上只存在一个事务,只是在调用a方法时设置了一个保存点,当a方法回滚时,实际上是回滚到保存点上,并且当外部事务提交时,内部事务才会提交,外部事务如果回滚,内部事务会跟着回滚。

  1. 内部事务发生异常时,外部事务catch异常后,内外两个事务都回滚,但是方法不抛出异常
@Transactionalpublic void testRollbackOnly() {try {      indexService.a();   } catch (ClassNotFoundException e) {// 加上这句代码      TransactionInterceptor.currentTransactionStatus().setRollbackOnly();   }}

❞通过显示的设置事务的状态为RollbackOnly。这样当提交事务时会进入下面这段代码

显示回滚

最大的区别在于处理回滚时第二个参数传入的是false,这意味着回滚是回滚是预期之中的,所以在处理完回滚后并不会抛出异常。

 

读写分离跟事务结合使用时的问题

读写分离一般有两种实现方式

  1. 配置多数据源
  2. 依赖中间件,如MyCat

如果是配置了多数据源的方式实现了读写分离,那么需要注意的是:「如果开启了一个读写事务,那么必须使用写节点」「如果是一个只读事务,那么可以使用读节点」如果是依赖于MyCat等中间件那么需要注意:「只要开启了事务,事务内的SQL都会使用写节点(依赖于具体中间件的实现,也有可能会允许使用读节点,具体策略需要自行跟DB团队确认)」基于上面的结论,我们在使用事务时应该更加谨慎,在没有必要开启事务时尽量不要开启。❝一般我们会在配置文件配置某些约定的方法名字前缀开启不同的事务(或者不开启),但现在随着注解事务的流行,好多开发人员(或者架构师)搭建框架的时候在service类上加上了@Transactional注解,导致整个类都是开启事务的,这样严重影响数据库执行的效率,更重要的是开发人员不重视、或者不知道在查询类的方法上面自己加上@Transactional(propagation=Propagation.NOT_SUPPORTED)就会导致,所有的查询方法实际并没有走从库,导致主库压力过大。❞其次,关于如果没有对只读事务做优化的话(优化意味着将只读事务路由到读节点),那么@Transactional注解中的readOnly属性就应该要慎用。我们使用readOnly的原本目的是为了将事务标记为只读,这样当MySQL服务端检测到是一个只读事务后就可以做优化,少分配一些资源(例如:只读事务不需要回滚,所以不需要分配undo log段)。但是当配置了读写分离后,可能会可能会导致只读事务内所有的SQL都被路由到了主库,读写分离也就失去了意义。

总结

本文为事务专栏最后一篇啦!这篇文章主要是总结了工作中事务相关的常见问题,想让大家少走点弯路!希望大家可以认真读完哦,有什么问题可以直接在后台私信我或者加我微信!这篇文章也是整个Spring系列的最后一篇文章,之后可能会出一篇源码阅读心得,跟大家聊聊如何学习源码。另外今年也给自己定了个小目标,就是完成SSM框架源码的阅读。目前来说Spring是完成,接下来就是SpringMVC跟MyBatis。在分析MyBatis前,会从JDBC源码出发,然后就是MyBatis对配置的解析、MyBatis执行流程、MyBatis的缓存、MyBatis的事务管理以及MyBatis的插件机制。在学习SpringMVC前,会从TomCat出发,先讲清楚TomCat的原理,我们再来看SpringMVC。整个来说相比于Spring源码,我觉得应该不算特别难。

有道无术,术可成;有术无道,止于术

欢迎大家关注Java之道公众号

好文章,我在看❤️

多方法调用 一个出错 集体回滚_一个@Transaction哪里来这么多坑?相关推荐

  1. mysql 回滚_一个集审核、执行、备份及生成回滚语句于一身的MySQL运维工具

    goInception 一个集审核.执行.备份及生成回滚语句于一身的MySQL运维工具, 通过对执行SQL的语法解析,返回基于自定义规则的审核结果,并提供执行和备份及生成回滚语句的功能 架构图 使用方 ...

  2. oracle一个循环中回滚继续,oracle回滚段

    http://hi.baidu.com/ipeipei/blog/item/34f84316f7126d4a20a4e950.html 1. 概述 本文主要从回滚段的原理,分配和使用,以及回滚段的相关 ...

  3. java 自定义异常 未回滚_抛出自定义异常,spring AOP事务不回滚的解决方案

    spring AOP 默认对RuntimeException()异常或是其子类进行事务回滚,也就是说 事务回滚:throw new RuntimeException("xxxxxxxxxxx ...

  4. git 代码回滚_能提交到远程的Git回滚

    很多情况下我们需要回滚代码,最容易想到的就是git reset.但是git reset有个弱点,它是一个彻底的回滚,不能再提交给远程了,因为在提交记录里回滚点之后的记录都不见了. 做一下试验,一个文件 ...

  5. dubbo调用超时回滚_如何处理Dubbo调用超时?

    原创:Java派(微信公众号:Java派),欢迎分享,转载请保留出处. 前言 Dubbo是阿里开源的RPC框架,因为他基于接口开发支持负载均衡.集群容错.版本控制等特性,因此现在有很多互联网公司都在使 ...

  6. dubbo调用超时回滚_微服务痛点基于Dubbo + Seata的分布式事务(AT模式)

    前言 Seata 是一款开源的分布式事务解决方案,致力于提供高性能和简单易用的分布式事务服务.Seata 将为用户提供了 AT.TCC.SAGA 和 XA 事务模式,为用户打造一站式的分布式解决方案. ...

  7. jpa transaction 回滚_我遇到的JPA中事务回滚的问题

    在最近的项目中,做的是解析XML文件,解析过程中会有异常,比如:XML文件中节点的数据和与之对应的数据库的字段中数据的类型不匹配:XML中数据长度超过数据库定义的长度:有数据了的重复插入问题:读取节点 ...

  8. springboot 事务手动回滚_来,讲讲Spring事务有哪些坑?

    来自公众号:孤独烟 引言 今天,我们接上文<面试官:谈谈你对mysql事务的认识>的内容,来讲spring中和事务有关的考题! 因为事务这块,面试的出现几率很高.而大家工作中CRUD的比较 ...

  9. 多个mapper的事务回滚_揭秘蚂蚁金服分布式事务 Seata 的AT、Saga和TCC模式

    作者| 屹远(陈龙),蚂蚁金服分布式事务核心研发 . 导语 本文根据 8月11日 SOFA Meetup#3 广州站 <分布式事务 Seata 及其三种模式详解>主题分享整理,着重分享分布 ...

最新文章

  1. pandas使用apply函数基于条件(if condition)生成新的数据列
  2. c# 重写override
  3. 2017年html5行业报告,云适配发布2017 HTML5开发者生态报告 期待更多行业标准
  4. JDK8新特性之方法引用
  5. OpenGL ARB 看来终于想通了,OpenGL SDK终于要出来了。
  6. 纯数学教程 Page 325 例LXVIII (15) 调和级数发散
  7. 安卓app测试之Monkey日志分析《转载》
  8. linux服务器MQ组件报警,服务器 有哪些告警
  9. 哈希表(散列表)冲突解决方法
  10. 让两个Div并排显示
  11. 普元云计算-你适合微服务么:实施微服务的4个先决条件和重点工作
  12. 发动机冒黑烟_发动机冒黑烟的原因和解决方法
  13. 文章最重要的并不是原创,而是伪原创
  14. 在Jetty中快速搭建SSL
  15. 经典散文·1.地毯的那一端
  16. linux mv覆盖目录,linux下利用grep和dd命令恢复被mv命令覆盖的文件内容
  17. iOS 分析一次有意思的需求——HTML代码注入
  18. Python求100以内素数的和。
  19. 段视频伪原创 手机如何去除视频md5
  20. 我的 gVim 初步配置

热门文章

  1. Phoenix 关联映射 Hbase表 获取不到数据,upsert hbase 列名为16进制字符
  2. 以系统化视角反观产品运营,解读提升用户转化的“四部曲”
  3. 可应用于实际的14个NLP突破性研究成果(一)
  4. 阿里云HBase发布冷存储特性,助你不改代码,1/3成本轻松搞定冷数据处理
  5. 新一代数据库技术在双11中的黑科技
  6. 1024 程序员节专题论坛来袭,聚焦企业级开源数据库 openGauss
  7. 谷歌这波操作,预警了什么信号??
  8. 那些被大数据时代抛弃的人
  9. 普惠数据科学应用,九章云极携手伙伴共探智慧未来
  10. AI时代的交换机什么样?华为CloudEngine 16800告诉你!