各位用户为了找寻关于由不同的索引更新解决MySQL死锁套路的资料费劲了很多周折。这里教程网为您整理了关于由不同的索引更新解决MySQL死锁套路的相关资料,仅供查阅,以下为您介绍关于由不同的索引更新解决MySQL死锁套路的详细内容
前几篇文章介绍了用源码的方式来调试锁相关的信息,这里同样用这个工具来解决一个线上实际的死锁案例,也是我们介绍的第一个两条 SQL 就造成死锁的情况。因为线上的表结构比较复杂,做了一些简化以后如下
? 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17CREATE
TABLE
`t3` (
`id`
int
(11)
NOT
NULL
AUTO_INCREMENT,
`a`
varchar
(5),
`b`
varchar
(5),
PRIMARY
KEY
(`id`),
UNIQUE
KEY
`uk_a` (`a`),
KEY
`idx_b` (`b`)
)
INSERT
INTO
`t3` (`id`, `a`, `b`)
VALUES
(1,
'1'
,
'2'
);
# sql语句如下
# 事务1:t1
update
t3
set
b =
''
where
a =
"1"
;
# 事务2:t2
update
t3
set
b =
''
where
b =
"2"
;
两条语句造成死锁的情况用手动的方式比较难复现,我们先来分析一下加锁的过程
第一条语句(通过唯一索引去更新记录)
? 1update
t3
set
b =
''
where
a =
"1"
;
整理一下,加了3个X锁,顺序分别是
第二条语句
? 1update
t3
set
b =
''
where
b =
"2"
;
整理一下,加了 3 个 X 锁,顺序分别是
两条语句从加锁顺序看起来就已经有构成死锁的条件了
手动是比较难模拟的,写个代码并发的去同时执行那两条 SQL 语句,马上就出现死锁了
? 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41------------------------
LATEST DETECTED DEADLOCK
------------------------
181102 12:45:05
*** (1)
TRANSACTION
:
TRANSACTION
50AF, ACTIVE 0 sec starting
index
read
mysql tables
in
use 1, locked 1
LOCK 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
update
update
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 waiting
Record 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
deleting
mysql tables
in
use 1, locked 1
4 lock struct(s), heap
size
1248, 3 row lock(s), undo log entries 1
MySQL thread id 35, OS thread handle 0x70000d885000, query id 548 localhost 127.0.0.1 root Updating
update
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
gap
Record 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 waiting
Record 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)
分析一下死锁日志
? 1 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 waiting
事务2:想获取主键索引的 X 锁
? 1 2*** (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
gap
事务1:持有主键索引的 X 锁
? 1 2*** (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 waiting
事务1:想获取普通索引 idx_b 的 X 锁
与我们分析的完全一致,也与线上的死锁日志一模一样
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。
原文链接:https://juejin.im/post/5ce3cfa46fb9a07ece67a4a9