成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

MySQL 死鎖套路:唯一索引下批量插入順序不一致

toddmark / 3469人閱讀

摘要:死鎖的本質(zhì)是資源競爭,批量插入如果順序不一致很容易導(dǎo)致死鎖,我們來分析一下這個(gè)情況。為了方便演示,把批量插入改寫為了多條。

死鎖的本質(zhì)是資源競爭,批量插入如果順序不一致很容易導(dǎo)致死鎖,我們來分析一下這個(gè)情況。為了方便演示,把批量插入改寫為了多條 insert。

先來做幾個(gè)小實(shí)驗(yàn),簡化的表結(jié)構(gòu)如下

CREATE TABLE `t1` (
  `id` int(11) NOT NULL AUTO_INCREMENT,
  `a` varchar(5),
  `b` varchar(5),
  PRIMARY KEY (`id`),
  UNIQUE KEY `uk_name` (`a`,`b`)
);
實(shí)驗(yàn)1:

在記錄不存在的情況下,兩個(gè)同樣順序的批量 insert 同時(shí)執(zhí)行,第二個(gè)會(huì)進(jìn)行鎖等待狀態(tài)

t1 t2
begin; begin;
insert ignore into t1(a, b)values("1", "1"); 成功
insert ignore into t1(a, b)values("1", "1"); 鎖等待狀態(tài)

可以看到目前鎖的狀態(tài)

mysql> select * from information_schema.innodb_locks;
+-------------+-------------+-----------+-----------+------------+------------+------------+-----------+----------+-----------+
| lock_id     | lock_trx_id | lock_mode | lock_type | lock_table | lock_index | lock_space | lock_page | lock_rec | lock_data |
+-------------+-------------+-----------+-----------+------------+------------+------------+-----------+----------+-----------+
| 31AE:54:4:2 | 31AE        | S         | RECORD    | `d1`.`t1`  | `uk_name`  |         54 |         4 |        2 | "1", "1"  |
| 31AD:54:4:2 | 31AD        | X         | RECORD    | `d1`.`t1`  | `uk_name`  |         54 |         4 |        2 | "1", "1"  |
+-------------+-------------+-----------+-----------+------------+------------+------------+-----------+----------+-----------+

在我們執(zhí)行事務(wù)t1的 insert 時(shí),沒有在任何鎖的斷點(diǎn)處出現(xiàn),這跟 MySQL 插入的原理有關(guān)系

insert 加的是隱式鎖。什么是隱式鎖?隱式鎖的意思就是沒有鎖

在 t1 插入記錄時(shí),是不加鎖的。這個(gè)時(shí)候事務(wù) t1 還未提交的情況下,事務(wù) t2 嘗試插入的時(shí)候,發(fā)現(xiàn)有這條記錄,t2 嘗試獲取 S 鎖,會(huì)判定記錄上的事務(wù) id 是否活躍,如果活躍的話,說明事務(wù)未結(jié)束,會(huì)幫 t1 把它的隱式鎖提升為顯式鎖( X 鎖)

源碼如下

t2 獲取S鎖的結(jié)果:DB_LOCK_WAIT

實(shí)驗(yàn)2:

批量插入順序不一致的導(dǎo)致的死鎖

t1 t2
begin
insert into t1(a, b)values("1", "1"); 成功
insert into t1(a, b)values("2", "2"); 成功
insert into t1(a, b)values("2", "2"); t1 嘗試獲取 S 鎖,把 t2 的隱式鎖提升為顯式 X 鎖,進(jìn)入 DB_LOCK_WAIT
insert into t1(a, b)values("1", "1"); t2 嘗試獲取 S 鎖,把 t1 的隱式鎖提升為顯式 X 鎖,產(chǎn)生死鎖
------------------------
LATEST DETECTED DEADLOCK
------------------------
181101  9:48:36
*** (1) TRANSACTION:
TRANSACTION 3309, ACTIVE 215 sec inserting
mysql tables in use 1, locked 1
LOCK WAIT 3 lock struct(s), heap size 376, 2 row lock(s), undo log entries 2
MySQL thread id 2, OS thread handle 0x70000a845000, query id 58 localhost root update
insert into t1(a, b)values("2", "2")
*** (1) WAITING FOR THIS LOCK TO BE GRANTED:
RECORD LOCKS space id 55 page no 4 n bits 72 index `uk_name` of table `d1`.`t1` trx id 3309 lock mode S waiting
Record lock, heap no 3 PHYSICAL RECORD: n_fields 3; compact format; info bits 0
 0: len 1; hex 32; asc 2;;
 1: len 1; hex 32; asc 2;;
 2: len 4; hex 80000002; asc     ;;

*** (2) TRANSACTION:
TRANSACTION 330A, ACTIVE 163 sec inserting
mysql tables in use 1, locked 1
3 lock struct(s), heap size 376, 2 row lock(s), undo log entries 2
MySQL thread id 3, OS thread handle 0x70000a888000, query id 59 localhost root update
insert into t1(a, b)values("1", "1")
*** (2) HOLDS THE LOCK(S):
RECORD LOCKS space id 55 page no 4 n bits 72 index `uk_name` of table `d1`.`t1` trx id 330A lock_mode X locks rec but not gap
Record lock, heap no 3 PHYSICAL RECORD: n_fields 3; compact format; info bits 0
 0: len 1; hex 32; asc 2;;
 1: len 1; hex 32; asc 2;;
 2: len 4; hex 80000002; asc     ;;

*** (2) WAITING FOR THIS LOCK TO BE GRANTED:
RECORD LOCKS space id 55 page no 4 n bits 72 index `uk_name` of table `d1`.`t1` trx id 330A lock mode S waiting
Record lock, heap no 2 PHYSICAL RECORD: n_fields 3; compact format; info bits 0
 0: len 1; hex 31; asc 1;;
 1: len 1; hex 31; asc 1;;
 2: len 4; hex 80000001; asc     ;;

*** WE ROLL BACK TRANSACTION (2)

怎么樣解決這樣的問題呢? 一個(gè)可行的辦法是在應(yīng)用層排序以后再插入

文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/7267.html

相關(guān)文章

  • MySQL 死鎖套路:一次詭異的批量插入死鎖問題分析

    摘要:線上最近出現(xiàn)了批量的死鎖,百思不得姐。死鎖記錄如下第一反應(yīng)是批量,的順序不一樣導(dǎo)致的死鎖。什么是隱式鎖隱式鎖的意思就是沒有鎖在插入記錄時(shí),是不加鎖的。線上最近出現(xiàn)了批量insert的死鎖,百思不得姐。死鎖記錄如下 2018-10-26T11:04:41.759589Z 8530809 [Note] InnoDB: *** (1) TRANSACTION: TRANSACTION 1202...

    afishhhhh 評(píng)論0 收藏0
  • 為什么開發(fā)人員必須要了解數(shù)據(jù)庫鎖?

    摘要:小明馬上開發(fā)完畢,成功上線。下班過后,小明回想大紅說的話,什么是間隙鎖,什么是插入意向鎖,看來作為開發(fā)者對(duì)數(shù)據(jù)庫不應(yīng)該只會(huì)寫啊,不然遇到一些疑難雜癥完全沒法解決啊。破壞了數(shù)據(jù)庫中的隔離性。 1.鎖? 1.1何為鎖 鎖在現(xiàn)實(shí)中的意義為:封閉的器物,以鑰匙或暗碼開啟。在計(jì)算機(jī)中的鎖一般用來管理對(duì)共享資源的并發(fā)訪問,比如我們java同學(xué)熟悉的Lock,synchronized等都是我們常見的...

    AbnerMing 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

最新活動(dòng)
閱讀需要支付1元查看
<