mysql中InnoDB事務(wù)隔離的記錄鎖、間隙鎖和臨鍵鎖
InnoDB默認(rèn)的事務(wù)隔離級別為可重復(fù)讀(Repeated Read, RR),我們當(dāng)下的所有介紹都是基于這個隔離級別為前提的。
- 記錄鎖(Record Locks):鎖定單一行記錄,InnoDB 使用記錄鎖來實現(xiàn)行級鎖,這樣允許多個事務(wù)并發(fā)訪問不同的行。
- 間隙鎖(Gap Locks):InnoDB 的特性,用于鎖定一個范圍,但不包括實際的記錄。這主要用于防止幻讀(Phantom Reads)。
- 臨鍵鎖(Next-Key Locks):InnoDB 存儲引擎的一種鎖定機(jī)制,在執(zhí)行查詢語句時,根據(jù)查詢條件所鎖定的一個范圍。這個范圍中包含有間隙鎖和記錄鎖。它的設(shè)計目的是為了解決幻讀(Phantom Reads)。
記錄鎖(Record Locks)
記錄鎖,它封鎖索引記錄,例如:
select * from table where id=5 for update;
它會在id=1的索引記錄上加鎖,以阻止其他事務(wù)插入,更新,刪除id=1的這一行。
需要說明的是:
select * from table where id=5;
則是快照讀(SnapShot Read),它并不加鎖,快照讀可以參考作者這篇文章:數(shù)據(jù)庫系列:RR和RC下,快照讀的區(qū)別
間隙鎖(Gap Locks)
間隙鎖,它封鎖索引記錄中的間隔,或者第一條索引記錄之前的范圍,又或者最后一條索引記錄之后的范圍。
延續(xù)上面的那個例子繼續(xù)演示:
# 表結(jié)構(gòu) table (Id PK, Name , Company); # 表中包含四條記錄 5, Gates, Microsoft 7, Bezos, Amazon 11, Jobs, Apple 14, Elison, Oracle
執(zhí)行SQL語句如下:
select * from table where id between 7 and 13 for update;
這樣的話,會封鎖數(shù)據(jù)的區(qū)間,以防止其他事務(wù)插入id=8的記錄。
假設(shè)沒有間隙鎖,則可能夠插入成功,而之前的select事務(wù),會發(fā)現(xiàn)檢索的結(jié)果集莫名多了一條記錄,即幻影數(shù)據(jù)。
所以間隙鎖主要目的用于防止幻讀(Phantom Reads),避免其他事務(wù)在間隔中插入數(shù)據(jù),導(dǎo)致 『不可重復(fù)讀』。
如果把事務(wù)的隔離級別降級為讀提交(Read Committed, RC),對,就是互聯(lián)網(wǎng)最常用的隔離級別,間隙鎖則會自動失效。
臨鍵鎖(Next-Key Locks)
臨鍵鎖(Next-Key Locks)是數(shù)據(jù)庫管理系統(tǒng)InnoDB中的一種重要鎖定機(jī)制。這種鎖是查詢時根據(jù)查詢條件鎖定的一個范圍,這個范圍包括間隙鎖和記錄鎖,左開右閉,即不鎖住左邊界,但會鎖住右邊界。臨鍵鎖的主要設(shè)計目的是為了解決所謂的“幻讀”問題。
# 左開右閉 示例 (-infinity, 1] (1, 7] (7, 9] (9, +infinity]
依然沿用上面的例子,InnoDB引擎,RR隔離級別:
-- 創(chuàng)建一個示例表 CREATE TABLE users ( Id INT PRIMARY KEY, Name VARCHAR(255) NOT NULL, Company VARCHAR(255) NOT NULL, ); -- 插入一些示例數(shù)據(jù) INSERT INTO users (id, name, company) VALUES (1, 'Alice', 'ali'); INSERT INTO users (id, name, company) VALUES (2, 'Brand', 'tencent'); INSERT INTO users (id, name, company) VALUES (3, 'Charlie', 'baidu'); -- 開始一個事務(wù),并使用臨鍵鎖查詢數(shù)據(jù) START TRANSACTION; SELECT * FROM users WHERE id > 1 FOR UPDATE; -- 在另一個事務(wù)中嘗試插入新數(shù)據(jù),將會被阻塞直到第一個事務(wù)釋放鎖 START TRANSACTION; INSERT INTO users (id, name, age) VALUES (4, 'David', 30); COMMIT; -- 第一個事務(wù)提交后,第二個事務(wù)可以繼續(xù)執(zhí)行插入操作 COMMIT;
臨鍵鎖的主要目的,也是為了避免幻讀(Phantom Read),在事務(wù)隔離級別為可重復(fù)讀的情況下,InnoDB存儲引擎默認(rèn)使用臨鍵鎖。這種鎖提供了一種有效的機(jī)制來保證在并發(fā)環(huán)境中數(shù)據(jù)的完整性和一致性。
如果把事務(wù)的隔離級別降級為RC,臨鍵鎖則也會失效。
總結(jié)
- InnoDB的索引與行記錄存儲在一起,MyISAM則是通過索引的地址查找到對應(yīng)的數(shù)據(jù)記錄,效率低一些
- InnoDB的聚集索引存儲行記錄,普通索引存儲PK,所以普通索引要查詢兩次
- 記錄鎖鎖定索引關(guān)聯(lián)的具體記錄
- 間隙鎖鎖定間隔,防止間隔中被其他事務(wù)插入
- 臨鍵鎖鎖定索引記錄+間隔,防止幻讀
- select...for update加鎖的幾種情況:
- 主鍵字段:加行鎖。
- 唯一索引字段:加行鎖。
- 普通索引字段:加行鎖。
- 主鍵范圍:加多個行鎖。
- 普通字段:加表鎖。
- 查詢空數(shù)據(jù):不加鎖。
- 行鎖與表鎖的區(qū)別
- 如果事務(wù)1加了行鎖,一直未釋放鎖,事務(wù)2操作相同記錄,會一直等待直至超時。
- 如果事務(wù)1加了表鎖,一直未釋放鎖,事務(wù)2無論操作哪一行記錄,都會一直等待直到超時
到此這篇關(guān)于mysql中InnoDB事務(wù)隔離的記錄鎖、間隙鎖和臨鍵鎖的文章就介紹到這了,更多相關(guān)InnoDB的記錄鎖、間隙鎖和臨鍵鎖內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
CentOS 6.5下yum安裝 MySQL-5.5全過程圖文教程
在linux安裝mysql是一個困難的事情,yum安裝一般是安裝的mysql5.1,現(xiàn)在經(jīng)過自己不懈努力終于能用yum安裝mysql5.5了。下面通過兩種方法給大家介紹CentOS 6.5下yum安裝 MySQL-5.5全過程,一起學(xué)習(xí)吧2016-05-05MySQL關(guān)聯(lián)查詢優(yōu)化實現(xiàn)方法詳解
在數(shù)據(jù)庫的設(shè)計中, 我們通常都是會有很多張表 , 通過表與表之間的關(guān)系建立我們想要的數(shù)據(jù)關(guān)系, 所以在多張表的前提下, 多表的關(guān)聯(lián)查詢就尤為重要,這篇文章主要介紹了MySQL關(guān)聯(lián)查詢優(yōu)化2022-11-11在MySQL中按字符串中的數(shù)字排序的詳細(xì)教程
本文將詳細(xì)介紹如何在MySQL中利用正則表達(dá)式提取字符串中的數(shù)字并按這些數(shù)字進(jìn)行排序,以一個具體的例子來說明,使得即使是數(shù)據(jù)庫操作的初學(xué)者也能輕松理解和應(yīng)用,需要的朋友可以參考下2024-07-07mysql中數(shù)據(jù)統(tǒng)計的技巧備忘錄
mysql是常用數(shù)據(jù)庫,對于數(shù)字操作相關(guān)的東西相當(dāng)方便,這篇文章主要給大家介紹了關(guān)于mysql中數(shù)據(jù)統(tǒng)計技巧的相關(guān)資料,非常具有實用價值,需要的朋友可以參考下2018-05-05