MySQL 死鎖套路:走不一樣的索引更新

前幾篇文章介紹了用源碼的方式來調試鎖相關的信息,這裏一樣用這個工具來解決一個線上實際的死鎖案例,也是咱們介紹的第一個兩條 SQL 就形成死鎖的狀況。由於線上的表結構比較複雜,作了一些簡化之後以下mysql

CREATE 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";
複製代碼

兩條語句形成死鎖的狀況用手動的方式比較難復現,咱們先來分析一下加鎖的過程sql

第一條語句(經過惟一索引去更新記錄)

update t3 set b = '' where a = "1";bash

整理一下,加了3個X鎖,順序分別是

序號 索引 鎖類型
1 uk_a X
2 PRIMARY X
3 idx_b X

第二條語句

update t3 set b = '' where b = "2";併發

整理一下,加了 3 個 X 鎖,順序分別是

序號 索引 鎖類型
1 idx_b X
2 PRIMARY X
3 idx_b X

兩條語句從加鎖順序看起來就已經有構成死鎖的條件了工具

手動是比較難模擬的,寫個代碼併發的去同時執行那兩條 SQL 語句,立刻就出現死鎖了ui

------------------------
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)
複製代碼

分析一下死鎖日誌spa

*** (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 waiting3d

事務2:想獲取主鍵索引的 X 鎖調試

*** (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 鎖

*** (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 鎖

與咱們分析的徹底一致,也與線上的死鎖日誌如出一轍

相關文章
相關標籤/搜索