欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

淺談Mysql insert on duplicate key 死鎖問題定位與解決

 更新時間:2022年05月05日 15:28:10   作者:呼呼虎  
本文介紹了在并發(fā)場景下的 insert on duplicate key update sql 出現(xiàn)的死鎖,經(jīng)過分析發(fā)現(xiàn)這種sql確實比較容易造成死鎖,這篇文章就從分析死鎖展開,到最終如何解決這樣的問題 分享相應(yīng)的思路,感興趣的可以了解一下

前言

最近在監(jiān)測線上日志時發(fā)現(xiàn)我們一個Mysql業(yè)務(wù)db時常出現(xiàn) dead lock,頻次不高但卻一直出現(xiàn),定位后發(fā)現(xiàn)是在并發(fā)場景下的 insert on duplicate key update sql 出現(xiàn)的死鎖。經(jīng)過分析發(fā)現(xiàn)這種sql確實比較容易造成死鎖,不太適用于我們目前的業(yè)務(wù)場景,于是更換后解決問題。

這篇文章就從分析死鎖展開,到最終如何解決這樣的問題 分享相應(yīng)的思路。

死鎖定位

我們目前生產(chǎn)環(huán)境使用Mysql版本為5.7,默認事務(wù)隔離級別為RR,以下為我們的大致table結(jié)構(gòu)(字段已經(jīng)完全脫敏,使用非業(yè)務(wù)字段)。

CREATE TABLE IF NOT EXISTS `user_info` (
        id BIGINT UNSIGNED NOT NULL AUTO_INCREMENT,
        name VARCHAR(20) NOT NULL,
        phone BIGINT(20) UNSIGNED NOT NULL,
        update_time timestamp  NOT NULL,
        UNIQUE KEY phone (phone)
 ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;

造成死鎖的sql如下:

insert into user_info (name, phone, update_time) values (X,Y,Z) on duplicate key update update_time=Z;

當(dāng)我們看到死鎖后,在對應(yīng)數(shù)據(jù)庫中進行分析,”show engine innodb status“,就發(fā)現(xiàn)這樣的報錯信息"lock_mode X locks gap before rec insert intention waiting"。意思就是在等待gap lock(間隙鎖)。

于是我們開始分析on duplicate key這個關(guān)鍵字的sql所可能引入的鎖,以及對應(yīng)我們業(yè)務(wù)場景中可能觸發(fā)死鎖的問題。

insert on duplicate key的鎖

首先insert on duplicate key 這條sql的語義是:如果insert中的對應(yīng)鍵值在數(shù)據(jù)庫中沒有找到對應(yīng)的唯一索引記錄,即進行插入;如果對表中唯一索引記錄沖突,便進行更新,能夠很輕松的達到一種效果: 有則直接更新,無則插入。而我們業(yè)務(wù)中的sql是自增主鍵id,這樣一來沖突的只有可能是 phone這個唯一索引了。

首先,在RR的事務(wù)隔離級別下,insert on duplicate key這個sql與普通insert只插入意向鎖和記錄鎖不同,insert on duplicate key sql如果沒有找到對應(yīng)的會在唯一鍵上插入gap lock和插入意向鎖(如果有對應(yīng)記錄則會獲取next key lock,next key lock 比gap lock多了一個邊緣的記錄鎖)。Mysql sql lock。

gap lock即間隙鎖,假設(shè)目前表中唯一鍵的數(shù)據(jù)有以下幾個,1,5,10。那么insert的key如果是4,在1-5之間,則獲取的gap lock的區(qū)間就是(1,5);如果插入的數(shù)據(jù)是15,則在10-正無窮之間,因此gap lock的區(qū)間就是(10,正無窮),這個gap lock。

插入意向鎖也是類似于gap lock的一種,生效的范圍也一致,只是對應(yīng)鎖上相同范圍或者有交集的。橫軸為已持有,縱軸為后續(xù)申請,是否互斥或兼容。

兼容性插入意向鎖行鎖gap lock
插入意向鎖兼容互斥互斥
行鎖兼容互斥兼容
gap lock兼容兼容兼容

因此可以看到,在持有g(shù)ap lock時,在插入的時候如果申請插入意向鎖,便會需要等待,而insert on duplicate key的sql在執(zhí)行時一般就是gap lock和插入意向鎖。那么造成死鎖的問題就定位到了,肯定是同一時間多個insert事務(wù)到來,并且所插入的記錄對應(yīng)的唯一鍵范圍基本一致,所擁有的gap lock和插入意向鎖的范圍有交集,便可以出現(xiàn)共同持有鎖反而造成死鎖的問題。

那我們大致還原一下對應(yīng)場景,以下是目前數(shù)據(jù)庫中的數(shù)據(jù)

idnamephonetimestamp
1jack155000000001970.1.1
2tom156000000001970.1.1
3hurry157000000001970.1.1
階段tx1tx2tx3
1insert into user_info (name, phone, update_time) values (test1,15700000001,1970.1.1) on duplicate key update update_time=now();  
1持有(15700000001,正無窮)的插入意向鎖以及gap lock  
2 insert into user_info (name, phone, update_time) values (test2,15700000002,1970.1.1) on duplicate key update update_time=now(); 
2 申請(15700000002,正無窮)的插入意向鎖失敗,申請gap lock成功,等待中 
3  insert into user_info (name, phone, update_time) values (test3,15700000004,1970.1.1) on duplicate key update update_time=now();
3  申請(15700000003,正無窮)的插入意向鎖失敗,申請gap lock成功,等待中
4commit 提交事務(wù),釋放鎖  
5 申請插入意向鎖成功申請插入意向鎖成功
6 死鎖死鎖

因此形成死鎖,其中一個事務(wù)回滾。

問題解決

可以看到,在我們的業(yè)務(wù)場景中,并沒有特別復(fù)雜的sql,但是仍然會導(dǎo)致死鎖,主要是插入數(shù)據(jù)的有序性以及高并發(fā)性,因此我們的解決思路也相對簡單。

針對我們業(yè)務(wù)的幾個思路:

  • 取消使用insert on duplicate key sql,換用普通insert sql,然后捕獲對應(yīng)dupicate 異常,進行異常重試和插入;
  • 業(yè)務(wù)上進行接口限流,并且入?yún)?shù)據(jù)的insert on duplicate key 數(shù)據(jù)list大小在事務(wù)中進行控制,分批執(zhí)行,可以減少死鎖的情況。

insert on duplicate key 雖然很方便一條sql完成幾條sql的事情,保證原子性,但是還是不適用于較高并發(fā)的場景,使用時需要多權(quán)衡。

到此這篇關(guān)于淺談Mysql insert on duplicate key 死鎖問題定位與解決的文章就介紹到這了,更多相關(guān)Mysql insert on duplicate key 死鎖內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • 淺談mysql雙層not exists查詢執(zhí)行流程

    淺談mysql雙層not exists查詢執(zhí)行流程

    本文主要介紹了淺談mysql雙層not?exists查詢執(zhí)行流程,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2023-06-06
  • MySQL查看和修改最大連接數(shù)的方法步驟

    MySQL查看和修改最大連接數(shù)的方法步驟

    使用MySQL 數(shù)據(jù)庫的站點,當(dāng)訪問連接數(shù)過多時,就會出現(xiàn) "Too many connections" 的錯誤,所以我們需要設(shè)置MySQL查看和修改最大連接數(shù),具有一定的參考價值,感興趣的可以了解一下
    2023-10-10
  • mysql Myisamchk小工具使用手冊

    mysql Myisamchk小工具使用手冊

    Myisamchk是MyISAM表維護的一個非常實用的工具。可以使用myisamchk實用程序來獲得有關(guān)數(shù)據(jù)庫表的信息或檢查、修復(fù)、優(yōu)化他們。myisamchk適用MyISAM表(對應(yīng).MYI和.MYD文件的表)。
    2008-10-10
  • Centos 7下使用RPM包安裝MySQL 5.7.9教程

    Centos 7下使用RPM包安裝MySQL 5.7.9教程

    這篇文章主要為大家詳細介紹了Centos 7下使用RPM包安裝MySQL 5.7.9的教程,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2017-05-05
  • mysql中json基礎(chǔ)查詢詳解(附圖文)

    mysql中json基礎(chǔ)查詢詳解(附圖文)

    MySQL提供了一些函數(shù)來對JSON數(shù)據(jù)進行操作,下面這篇文章主要給大家介紹了關(guān)于mysql中json基礎(chǔ)查詢的相關(guān)資料,文中通過圖文以及實例代碼介紹的非常詳細,需要的朋友可以參考下
    2023-10-10
  • 關(guān)于SqlServer中datediff用法

    關(guān)于SqlServer中datediff用法

    datediff是SQL SERVER里面的用法,ORACLE沒有,主要作用是返回兩個日期之間的時間間隔,本文通過實例代碼給大家詳細講解,對datediff用法感興趣的朋友跟隨小編一起看看吧
    2022-11-11
  • MYSQL設(shè)置字段自動獲取當(dāng)前時間的sql語句

    MYSQL設(shè)置字段自動獲取當(dāng)前時間的sql語句

    整理數(shù)據(jù)庫數(shù)據(jù),看到好多表都有加create_time和 update_time字段,來記錄數(shù)據(jù)插入的時間和更新時間,但是時間插入是通過代碼來維護的,這篇文章主要介紹了MYSQL設(shè)置字段自動獲取當(dāng)前時間,需要的朋友可以參考下
    2023-07-07
  • Mysql之組合索引方法詳解

    Mysql之組合索引方法詳解

    這篇文章主要介紹了Mysql之組合索引方法詳解,文中通過示例代碼和查詢結(jié)果展示介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-08-08
  • 在同一臺機器上運行多個 MySQL 服務(wù)

    在同一臺機器上運行多個 MySQL 服務(wù)

    在同一臺機器上運行多個 MySQL 服務(wù)...
    2006-11-11
  • MySQL使用select語句查詢指定表中指定列(字段)的數(shù)據(jù)

    MySQL使用select語句查詢指定表中指定列(字段)的數(shù)據(jù)

    本文介紹MySQL數(shù)據(jù)庫中執(zhí)行select查詢語句,查詢指定列的數(shù)據(jù),即指定字段的數(shù)據(jù),需要的朋友可以參考下
    2016-11-11

最新評論