写点什么

MySQL 死锁套路:再来看一例走不同索引更新的例子

用户头像
AI乔治
关注
发布于: 2021 年 04 月 19 日
MySQL 死锁套路:再来看一例走不同索引更新的例子

前面有文章介绍了利用调试 MySQL 源码的方式来调试锁相关的信息,这里利用这个工具来解决一个比较简单的问题,线上的表字段较多,这里简单成为了一个表:

CREATE TABLE `t3` (  `id` int(11) NOT NULL AUTO_INCREMENT,  `a` varchar(5) COLLATE utf8mb4_unicode_ci NOT NULL DEFAULT '',  `b` varchar(5) COLLATE utf8mb4_unicode_ci NOT NULL DEFAULT '',  PRIMARY KEY (`id`),UNIQUE KEY `uk_a` (`a`), KEY `idx_b` (`b`) )
复制代码

sql 语句如下

t1update t3 set b = '' where a = "1";
t2update t3 set b = '' where b = "2";
复制代码

我们先用之前 debug 的方式来看一下,这里两条语句分别加了哪些锁

第一条语句(通过唯一索引去更新记录)

update t3 set b = ‘’ where a = “1”;



整理一下,加了 3 个 X 锁,顺序分别是


第二条语句

update t3 set b = ‘’ where b = “2”;




整理一下,加了 3 个 X 锁,顺序分别是

两条语句从加锁顺序看起来就已经有构成死锁的条件了


手动是比较难模拟的,写个代码去跑一下马上就出现了

------------------------LATEST DETECTED DEADLOCK------------------------181102 12:45:05*** (1) TRANSACTION:TRANSACTION 50AF, ACTIVE 0 sec starting index readmysql tables in use 1, locked 1LOCK WAIT 3 lock struct(s), heap size 376, 2 row lock(s)MySQL thread id 34, OS thread handle 0x70000d842000, query id 549 localhost 127.0.0.1 root Searching rows for updateupdate t3 set b = '' where b = "2"*** (1) WAITING FOR THIS LOCK TO BE GRANTED:RECORD LOCKS space id 67 page no 3 n bits 72 index `PRIMARY` of table `d1`.`t3` trx id 50AF lock_mode X locks rec but not gap waitingRecord lock, heap no 2 PHYSICAL RECORD: n_fields 5; compact format; info bits 0 0: len 4; hex 80000001; asc     ;; 1: len 6; hex 0000000050ae; asc     P ;; 2: len 7; hex 03000001341003; asc     4  ;; 3: len 1; hex 31; asc 1;; 4: len 0; hex ; asc ;;
*** (2) TRANSACTION:TRANSACTION 50AE, ACTIVE 0 sec updating or deletingmysql tables in use 1, locked 14 lock struct(s), heap size 1248, 3 row lock(s), undo log entries 1MySQL thread id 35, OS thread handle 0x70000d885000, query id 548 localhost 127.0.0.1 root Updatingupdate t3 set b = '' where a = "1"*** (2) HOLDS THE LOCK(S):RECORD LOCKS space id 67 page no 3 n bits 72 index `PRIMARY` of table `d1`.`t3` trx id 50AE lock_mode X locks rec but not gapRecord lock, heap no 2 PHYSICAL RECORD: n_fields 5; compact format; info bits 0 0: len 4; hex 80000001; asc ;; 1: len 6; hex 0000000050ae; asc P ;; 2: len 7; hex 03000001341003; asc 4 ;; 3: len 1; hex 31; asc 1;; 4: len 0; hex ; asc ;;
*** (2) WAITING FOR THIS LOCK TO BE GRANTED:RECORD LOCKS space id 67 page no 5 n bits 72 index `idx_b` of table `d1`.`t3` trx id 50AE lock_mode X locks rec but not gap waitingRecord lock, heap no 2 PHYSICAL RECORD: n_fields 2; compact format; info bits 0 0: len 1; hex 32; asc 2;; 1: len 4; hex 80000001; asc ;;
*** WE ROLL BACK TRANSACTION (1)
复制代码

跟我们线上的死锁日志一模一样~


看完三件事❤️

  • 如果你觉得这篇内容对你还蛮有帮助,我想邀请你帮我三个小忙:

  • 点赞,转发,有你们的 『点赞和评论』,才是我创造的动力。

  • 关注公众号 『 java 烂猪皮 』,不定期分享原创知识。

  • 同时可以期待后续文章 ing🚀



作者:挖坑的张师傅

出处:https://club.perfma.com/article/2344089


用户头像

AI乔治

关注

分享后端技术干货。公众号【 Java烂猪皮】 2019.06.30 加入

一名默默无闻的扫地僧!

评论

发布
暂无评论
MySQL 死锁套路:再来看一例走不同索引更新的例子