两个小工具,MySQL死锁分析,新技能又Get!!!

本文涉及的产品
RDS AI 助手,专业版
RDS MySQL DuckDB 分析主实例,集群系列 4核8GB
RDS MySQL DuckDB 分析主实例,基础系列 4核8GB
简介: 死锁是MySQL中非常难调试的问题,常见的思路与方法有:(1)通过多终端模拟并发事务,复现死锁;(2)通过show engine innodb status; 可以查看事务与锁的信息;(3)通过explain可以查看执行计划;

数据库死锁,是最难调试与追踪的。

场景如下:
image.png

同一个表,事务内先插入一条记录,再更新这条记录,并发时会死锁。
image.png

并且能够复现。

可以通过什么工具模拟并发事务,查看信息,解决问题呢?这是今天要分享的内容。


一、前置准备

set session transaction isolation level repeatable read;

set session autocommit=0;

 

create table t (

id int(20) primary key AUTO_INCREMENT,

cell varchar(20) unique

)engine=innodb;

 

start transaction;

insert into t(cell) values(11111111111);

insert into t(cell) values(22222222222);

insert into t(cell) values(33333333333);

commit;

 

说明:

(1)案发时,事务隔离级别RR

(2)多终端实验,需要关闭事务自动提交

(3)建表,设置PK与unique,初始化数据;

 

二、并发事务模拟

Session A:

start transaction;

insert into t(cell)values(44444444444);  [1]

        Session B:

        start transaction;

        insert into t(cell) values(55555555555); [2]

update t set cell=123 where cell=44444444444; [3]   

        update t set cell=456 where cell=55555555555; [4]

 

开启两个终端模拟并发事务:

(1)红色SQL为事务A;

(2)黑色SQL为事务B;

(3)13为执行时序;


三、实验现象

insert into t(cell)values(44444444444);  [1]

事务A插入数据,最先执行

结果:插入成功

 

insert into t(cell) values(55555555555); [2]

事务B插入数据,第二执行

结果:插入成果

 

update t set cell=123 where cell=44444444444; [3]

事务A修改[1]中插入的数据,第三执行

结果:阻塞,等待执行结果
image.png

画外音:修改一条自己插入的数据,在等待什么呢?

 

update t set cell=456 where cell=55555555555; [4]

事务B修改[2]中插入的数据,最后执行

结果:

(1)事务B死锁,事务B被回滚
image.png

(2)事务A中,[3]语句阻塞结束,执行成功
image.png

画外音:说明事务A中阻塞的语句,确实在等事务B中的某个锁。

 

四、结果分析

两个事务,各自修改自己插入的数据,却产生了死锁,确实诡异。


上述实验现象的两个核心问题是:

(1)语句[3]阻塞,在等待什么锁

(2)语句[4]死锁,此时事务A和事务B一定是彼此占住一把锁,请求彼此的锁,这些锁又是什么呢

 

工具一:

show engine innodb status;

画外音:前文《超赞,InnoDB调试死锁的方法!》就详细分享过,InnoDB死锁的分析实践。

 

执行之后,显示的内容如下(放大仔细看):
image.png

信息很多,别急,楼主娓娓道来。

 

第一部分,关键词是:

(1)Transaction 1,事务3998;

(2)在执行

update t set cell=123 where cell=44444444444;

(3)正在等待锁释放(waiting for this lock to be granted),记录锁(record locks),主键索引上(index primary),互斥锁(lock_mode X),物理记录(physical record),asc 55555555555;

画外音:英文比较差没事,抓关键词。

 

画外音,InnoDB存储引擎,聚集索引与非聚集索引的实现方式,决定了锁会加在聚集索引上。

第二部分,关键词是:

(1)Transaction 2,事务3999;

(2)正在执行

update t set cell=456 where cell=55555555555;

(3)持有锁(holds the lock),记录锁(record locks),主键索引上(index primary),互斥锁(lock_mode X),物理记录(physical record),asc 55555555555;

(4)正在等待锁释放(waiting for this lock to be granted),记录锁(record locks),主键索引上(index primary),互斥锁(lock_mode X),物理记录(physical record),asc 11111111111;

(5)事务2回滚(we roll back transaction 2)

 

通过show engine innodb status; 能够看到很多事务与锁之间的信息,对分析问题十分有帮助,这些信息,能够解释一些问题,但仍有两个疑惑:

(1)事务1为啥想拿55555555555的锁?

画外音:这正是,事务1被阻塞的原因。

 

(2)事务2为啥想拿11111111111的锁?死锁的发生,说明事务1此时真占着11111111111的锁,这又是为什么呢?

画外音:第一个事务占111抢555,第二个事务占555抢111,循环嵌套,才会死锁。

 

工具二:

explain

为了进一步寻找原因,可以通过explain看下导致死锁语句的执行计划。

 

explain update t set cell=456 where cell=55555555555;
image.png

:SIMPLE

这是一个简单类型的SQL语句,不含子查询或者UNION。

 

type:index

访问类型,即找到所需数据使用的遍历方式,潜在的方式有:

(1)ALL(Full Table Scan):全表扫描;

(2)index:走索引的全表扫描;

(3)range:命中where子句的范围索引扫描;

(4)ref/eq_ref:非唯一索引/唯一索引单值扫描;

(5)const/system:常量扫描;

(6)NULL:不用访问表;

 

上述扫描方式,ALL最慢,逐步变快,NULL最快。

 

怀疑点1:明明cell字段有uniq索引,为何要进行走PK索引的全表扫描呢?

 

possible_keys:NULL

可能在哪个索引找到记录。

 

key:PRIMARY

实际使用索引。

画外音:使用PK进行的全表扫描。

 

ref:NULL

哪些列,或者常量用于查找索引上的值。

 

怀疑点2:where条件中的查询条件55555555555,本来应该作为在索引上被检索的值呀?

 

rows:5

找到所需记录,预估需要读取的行数。

 

怀疑点3:明明修改的是5,为何初始化的1,2,3,以及第一个事务插入的4,以及第二个事务插入的5,都要被读取呢?不应该全表扫描呀。

 

通过explain,基本已经可以判断:

update t set cell=456 where cell=55555555555;

并没有和我们预想一样,走cell索引进行查询,而是走了PK索引进行了全表扫描

 

再仔细一看:

create table t (

id int(20) primary key AUTO_INCREMENT,

cell varchar(20) unique

)engine=innodb;

建表的时候cell定义的是字符串类型

 

更新的时候,

update t set cell=456 where cell=55555555555;

使用的是整数类型

 

类型转换,会导致全表扫描,出现锁升级,锁住全部记录

 

加上引号,再次通过explain验证一下:

explain update t set cell= '456 ' where cell= '55555555555 ';


image.png

果然印证了猜想:

(1)type:range,变为了走索引的字符串比对,范围扫描;

(2)possible_keys:cell,通过cell索引找到了记录;

(3)key:cell,实际使用cell索引;

(4)ref:const,使用了常量' 555'进行比对;

(5)rows:1,预估读取行数是1;

 

这下全部可以解释了
image.png

总结

就本例而言:需要注意字符串与整数之间的强制类型转换,有时候少一个引号,就会使得行锁升级为表锁。

 

死锁是MySQL中非常难调试的问题,常见的思路与方法有:

(1)通过多终端模拟并发事务,复现死锁;

(2)通过show engine innodb status; 可以查看事务与锁的信息;

(3)通过explain可以查看执行计划;

 

思路比结论更重要,希望大家有收获。

本文转自“架构师之路”公众号,58沈剑提供。

相关实践学习
自建数据库迁移到云数据库
本场景将引导您将网站的自建数据库平滑迁移至云数据库RDS。通过使用RDS,您可以获得稳定、可靠和安全的企业级数据库服务,可以更加专注于发展核心业务,无需过多担心数据库的管理和维护。
MySQL数据库入门学习
本课程通过最流行的开源数据库MySQL带你了解数据库的世界。   相关的阿里云产品:云数据库RDS MySQL 版 阿里云关系型数据库RDS(Relational Database Service)是一种稳定可靠、可弹性伸缩的在线数据库服务,提供容灾、备份、恢复、迁移等方面的全套解决方案,彻底解决数据库运维的烦恼。 了解产品详情: https://www.aliyun.com/product/rds/mysql 
目录
相关文章
|
7月前
|
缓存 关系型数据库 BI
使用MYSQL Report分析数据库性能(下)
使用MYSQL Report分析数据库性能
485 158
|
6月前
|
存储 消息中间件 监控
MySQL 到 ClickHouse 明细分析链路改造:数据校验、补偿与延迟治理
蒋星熠Jaxonic,数据领域技术深耕者。擅长MySQL到ClickHouse链路改造,精通实时同步、数据校验与延迟治理,致力于构建高性能、高一致性的数据架构体系。
MySQL 到 ClickHouse 明细分析链路改造:数据校验、补偿与延迟治理
|
7月前
|
缓存 监控 关系型数据库
使用MYSQL Report分析数据库性能(中)
使用MYSQL Report分析数据库性能
505 156
|
7月前
|
缓存 监控 关系型数据库
使用MYSQL Report分析数据库性能(上)
最终建议:当前系统是完美的读密集型负载模型,优化重点应放在减少行读取量和提高数据定位效率。通过索引优化、分区策略和内存缓存,预期可降低30%的CPU负载,同时保持100%的缓冲池命中率。建议每百万次查询后刷新统计信息以持续优化
608 161
|
6月前
|
NoSQL 算法 Redis
【Docker】(3)学习Docker中 镜像与容器数据卷、映射关系!手把手带你安装 MySql主从同步 和 Redis三主三从集群!并且进行主从切换与扩容操作,还有分析 哈希分区 等知识点!
Union文件系统(UnionFS)是一种**分层、轻量级并且高性能的文件系统**,它支持对文件系统的修改作为一次提交来一层层的叠加,同时可以将不同目录挂载到同一个虚拟文件系统下(unite several directories into a single virtual filesystem) Union 文件系统是 Docker 镜像的基础。 镜像可以通过分层来进行继承,基于基础镜像(没有父镜像),可以制作各种具体的应用镜像。
721 6
|
8月前
|
存储 关系型数据库 MySQL
深入理解MySQL索引类型及其应用场景分析。
通过以上介绍可以看出各类MySQL指标各自拥有明显利弊与最佳实践情墁,在实际业务处理过程中选择正确型号极其重要以确保系统运作流畅而稳健。
249 12
|
8月前
|
存储 关系型数据库 MySQL
在CentOS 8.x上安装Percona Xtrabackup工具备份MySQL数据步骤。
以上就是在CentOS8.x上通过Perconaxtabbackup工具对Mysql进行高效率、高可靠性、无锁定影响地实现在线快速全量及增加式数据库资料保存与恢复流程。通过以上流程可以有效地将Mysql相关资料按需求完成定期或不定期地保存与灾难恢复需求。
595 10
|
9月前
|
关系型数据库 MySQL Java
字节面试: MySQL 百万级 导入发生的 “死锁” 难题如何解决?“2序4拆”,彻底攻克
字节面试: MySQL 百万级 导入发生的 “死锁” 难题如何解决?“2序4拆”,彻底攻克
字节面试: MySQL 百万级 导入发生的 “死锁” 难题如何解决?“2序4拆”,彻底攻克
|
9月前
|
存储 SQL 关系型数据库
MySQL的Redo Log与Binlog机制对照分析
通过合理的配置和细致的管理,这两种日志机制相互配合,能够有效地提升MySQL数据库的可靠性和稳定性。
285 10
|
9月前
|
SQL 关系型数据库 MySQL
MySQL group by 底层原理详解。group by 执行 慢 原因深度分析。(图解+秒懂+史上最全)
MySQL group by 底层原理详解。group by 执行 慢 原因深度分析。(图解+秒懂+史上最全)
MySQL group by 底层原理详解。group by 执行 慢 原因深度分析。(图解+秒懂+史上最全)

推荐镜像

更多