Spring编程式事务使用不当导致其他事务无法正常提交

1.事故背景

原本在使用的是注解式事务,后面因为需要在事务中增加异步推送机制,所以需要将推送机制放到事务之外,修改后发现系统经常出现事务长时间无法提交导致回滚。

2.排查流程

(1)一开始重启应用是能恢复正常,所以肯定是在某种情况下会触发异常的产生

(2)查看在mysql控制台查看当前正在执行的事务(SELECT * FROM information_schema.INNODB_TRX),分析该sql语句在逻辑上并没有锁竞争的出现,只是单单一条update语句,但事务却没有提交

(3)这时候确定在业务代码逻辑上不会出现锁竞争,但事务却没有正常提交,所以考虑是mysql连接会话的autoCommit属性为false导致事务无法正常提交

(4)因异常出现是在讲注解式事务改为编程式事务之后,所以猜测是因为该改动导致异常出现

3.原理分析

(1)spring事务支持原理:spring的事务支持原理是先将mysql连接会话的自动提交属性关闭,即将当前会话的autoCommit属性设置为false,然后将该连接绑定到该线程中,在该事务中的所有数据库操作都是使用同一个线程,所有的数据库操作完成后才主动去做commit操作完成事务

(2)以下为编程式事务出现异常的流程分析的代码示例

当开启事务时,当前会话的自动提交属性讲被设置为false

 当事务没有提交而是提前return出去时,会话的状态并不会改变,autoCommit属性一直为false,这就导致当其他请求使用该数据库连接会话操作数据库时,事务将无法自动提交,如下图所示,即使没有开启事务,

autoCommit状态也是为false,所以会导致其他使用该会话的事务无法正常提交

 

 

 4.改进方式

(1)避免在未主动commit事务前return出去

(2)增加finally代码块,判断事务状态,回滚事务即可

DefaultTransactionDefinition defaultTransactionDefinition = new DefaultTransactionDefinition();
        TransactionStatus transactionStatus = transactionManager.getTransaction(defaultTransactionDefinition);
        try {
           doSomething();
        } catch (Exception e) {
            transactionManager.rollback(transactionStatus);
            getLogger().error(e.getMessage());
            throw new RuntimeException("系统异常");
        } finally {
            if(null != transactionStatus && !transactionStatus.isCompleted()){
                transactionManager.rollback(transactionStatus);
            }
        }

 连接恢复源码如下:

 

 



posted @ 2020-08-24 11:20  苹果大大个  阅读(1429)  评论(0编辑  收藏  举报