benym的知识笔记 benym的知识笔记
🦮首页
  • Java

    • Java-基础
    • Java-集合
    • Java-多线程与并发
    • Java-JVM
    • Java-IO
  • Python

    • Python-基础
    • Python-机器学习
  • Kafka
  • Redis
  • MySQL
  • 分布式事务
  • Spring

    • SpringIOC
    • SpringAOP
🦌设计模式
  • 剑指Offer
  • LeetCode
  • 排序算法
🐧实践
  • Rpamis

    • Utils
    • Exception
    • Security
  • 归档
  • 标签
  • 目录
🦉里程碑
🐷关于
GitHub (opens new window)

benym

惟其艰难,才更显勇毅🍂惟其笃行,才弥足珍贵
🦮首页
  • Java

    • Java-基础
    • Java-集合
    • Java-多线程与并发
    • Java-JVM
    • Java-IO
  • Python

    • Python-基础
    • Python-机器学习
  • Kafka
  • Redis
  • MySQL
  • 分布式事务
  • Spring

    • SpringIOC
    • SpringAOP
🦌设计模式
  • 剑指Offer
  • LeetCode
  • 排序算法
🐧实践
  • Rpamis

    • Utils
    • Exception
    • Security
  • 归档
  • 标签
  • 目录
🦉里程碑
🐷关于
GitHub (opens new window)
  • 站点优化

    • 将hexo自定义域名升级https
    • hexo到Typecho的迁移日志
  • 思考与方案

    • 海量数据TopK问题
    • 关于DO,VO,DTO,QueryParam的思考
    • 异步消息通知—异步改造
    • 二叉搜索树及AVL树详解
    • 简单高效的代码优化-事务后异步处理
    • 接口管理平台Yapi-最佳实践
    • Yapi私有化部署方案
    • Sentinel-Dashboard持久化生产环境解决方案
    • 单测覆盖率工具在多模块项目中的集成
    • DSTransactional与Transactional事务混用死锁场景分析
      • 问题背景
      • 问题流程图
      • 问题现象
      • 问题原因
      • 排查思路
      • 解决方案
  • AI人工智能

    • 基于Docker如何快速部署自己的ChatGPT
  • 实用代码

    • 编程式事务工具类
    • EasyExcel工具类
    • 本地锁工具类
    • Jackson基本配置类
    • Mybatis-plus基本配置类
    • RestTemplate基本配置类
    • 线程池基本配置类
    • RedisTemplate基本配置类
    • SpringData-Mongo基本配置类
    • SpringCache基本配置类
  • 实践
  • 思考与方案
benym
2024-03-04
目录

DSTransactional与Transactional事务混用死锁场景分析

# 问题背景

最近在生产环境发现了死锁问题,经过排查发现是由于在使用@DSTransactional跨数据源注解时,混合使用了@Transactional注解,造成了同时对某个表中数据行的更新,导致了死锁。以此记录下排查的过程

# 问题流程图

问题流程图如下

# 问题现象

生产环境偶现死锁异常日志

部分业务执行成功一半,事务没有正常回滚

# 问题原因

如流程图所示 如果你的代码包含如下内容,那么就会造成死锁

@Autowired
private OtherDao otherDao;

@DSTransactional
public String testTransactional(Test test){
    // do business
    testMapper.updateStatus(test);
    otherDao.revoke(test);
    return "";
}

@Mapper
public class OtherDao implements IOtherDao{
    
    @Transactional
    public void revoke(Test test) {
        testMapper.update(test);
    }
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19

这段代码描述了流程图中的事务顺序,先开启@DSTransactional,然后操作某一数据行,之后再开启@Transactional,操作同一数据行,由于@DSTransactional事务没有提交,等价于存在两个事务同时操作同一张表的同一行,产生了竞态,导致死锁发生

# 排查思路

1、找DBA要死锁日志找到死锁原因,或者有权限的情况下show engine innodb status\G,找到LATEST DETECTED DEADLOCK,再分析日志

2、根据日志排查代码,通常是因为事务的错误使用引起的

3、show processlist,查看当前执行情况,再来看代码

4、分析代码中是否有select读写锁,比如share mode或for update,update更新同一行等可疑情况

# 解决方案

避免混合使用@DSTransactional和@Transactional,因为@DSTransactional不支持事务传播机制,同时需要避免长事务

编辑 (opens new window)
#死锁#DSTransactional#Transactional#事务
上次更新: 2024/03/04, 18:21:06
单测覆盖率工具在多模块项目中的集成
基于Docker如何快速部署自己的ChatGPT

← 单测覆盖率工具在多模块项目中的集成 基于Docker如何快速部署自己的ChatGPT→

最近更新
01
SpringCache基本配置类
05-16
02
Rpamis-security-原理解析
12-13
03
Rpamis-security-技术背景
11-29
更多文章>
Theme by Vdoing | Copyright © 2018-2024 benym | MIT License
 |   |   | 
渝ICP备18012574号 | 渝公网安备50010902502537号
  • 跟随系统
  • 浅色模式
  • 深色模式
  • 阅读模式