MySQL數(shù)據(jù)庫(kù)InnoDB引擎行級(jí)鎖鎖定范圍詳細(xì)說明
發(fā)表時(shí)間:2023-07-14 來源:明輝站整理相關(guān)軟件相關(guān)文章人氣:
[摘要]Mysql數(shù)據(jù)庫(kù)InnoDB引擎支持行級(jí)鎖,也就是說我們可以對(duì)表中某些行數(shù)據(jù)執(zhí)行鎖定操作,鎖定操作的影響是:如果一個(gè)事物對(duì)表中某行執(zhí)行了鎖定操作,而另一個(gè)事務(wù)也需要對(duì)同樣的行執(zhí)行鎖定操作,這樣第二個(gè)...
Mysql數(shù)據(jù)庫(kù)InnoDB引擎支持行級(jí)鎖,也就是說我們可以對(duì)表中某些行數(shù)據(jù)執(zhí)行鎖定操作,鎖定操作的影響是:如果一個(gè)事物對(duì)表中某行執(zhí)行了鎖定操作,而另一個(gè)事務(wù)也需要對(duì)同樣的行執(zhí)行鎖定操作,這樣第二個(gè)事務(wù)的鎖定操作有可能被阻塞,一旦被阻塞第二個(gè)事務(wù)只能等到第一個(gè)事務(wù)執(zhí)行完畢(提交或回滾)或超時(shí)。
本文主要介紹InnoDB中的行鎖相關(guān)概念,重點(diǎn)介紹行鎖的鎖定范圍:
什么樣的SQL語(yǔ)句會(huì)加鎖?
加什么樣的鎖?
加鎖語(yǔ)句會(huì)鎖定哪些行?
背景知識(shí)
上面我們簡(jiǎn)單的介紹了InnoDB的行級(jí)鎖,為了理解后面的驗(yàn)證部分,需要補(bǔ)充一下背景知識(shí)。如果對(duì)相應(yīng)知識(shí)非常了解,可以直接跳轉(zhuǎn)到驗(yàn)證部分內(nèi)容。
1. InnoDB鎖的類型
InnoDB引擎使用了七種類型的鎖,他們分別是:
本文主要涉及Shared and Exclusive Locks,Record Locks,Gap Locks,Next-Key Locks這幾種鎖,其他類型鎖如果大家感興趣可以自己深入了解,在此不在詳述。
1.1 Shared and Exclusive Locks
共享鎖(S鎖)和排他鎖(X鎖)的概念在許多編程語(yǔ)言中都出現(xiàn)過。先來描述一下這兩種鎖在MySQL中的影響結(jié)果:
用一張經(jīng)典的矩陣表格繼續(xù)說明共享鎖和排他鎖的互斥關(guān)系:
圖中S表示共享鎖X表示獨(dú)占鎖,0表示鎖兼容1表示鎖沖突,兼容不被阻塞,沖突被阻塞。由表可知一旦一個(gè)事務(wù)加了排他鎖,其他個(gè)事務(wù)加任何鎖都需要等待。多個(gè)共享鎖不會(huì)相互阻塞。
1.2 Record Locks、Gap Locks、Next-Key Locks
這三種類型的鎖都描述了鎖定的范圍,故放在一起說明。
以下定義摘自MySQL官方文檔
記錄鎖(Record Locks):記錄鎖鎖定索引中一條記錄。
間隙鎖(Gap Locks):間隙鎖要么鎖住索引記錄中間的值,要么鎖住第一個(gè)索引記錄前面的值或者最后一個(gè)索引記錄后面的值。
Next-Key Locks:Next-Key鎖是索引記錄上的記錄鎖和在索引記錄之前的間隙鎖的組合。
定義中都提到了索引記錄(index record)。為什么?行鎖和索引有什么關(guān)系呢?其實(shí),InnoDB是通過搜索或者掃描表中索引來完成加鎖操作,InnoDB會(huì)為他遇到的每一個(gè)索引數(shù)據(jù)加上共享鎖或排他鎖。所以我們可以稱行級(jí)鎖(row-level locks)為索引記錄鎖(index-record locks),因?yàn)樾屑?jí)鎖是添加到行對(duì)應(yīng)的索引上的。
三種類型鎖的鎖定范圍不同,且逐漸擴(kuò)大。我們來舉一個(gè)例子來簡(jiǎn)要說明各種鎖的鎖定范圍,假設(shè)表t中索引列有3、5、8、9四個(gè)數(shù)字值,根據(jù)官方文檔的確定三種鎖的鎖定范圍如下:
記錄鎖的鎖定范圍是單獨(dú)的索引記錄,就是3、5、8、9這四行數(shù)據(jù)。
間隙鎖的鎖定為行中間隙,用集合表示為(-∞,3)、(3,5)、(5,8)、(8,9)、(9,+∞)。
Next-Key鎖是有索引記錄鎖加上索引記錄鎖之前的間隙鎖組合而成,用集合的方式表示為(-∞,3]、(3,5]、(5,8]、(8,9]、(9,+∞)。
最后對(duì)于間隙鎖還需要補(bǔ)充三點(diǎn):
間隙鎖阻止其他事務(wù)對(duì)間隙數(shù)據(jù)的并發(fā)插入,這樣可有有效的解決幻讀問題(Phantom Problem)。正因?yàn)槿绱耍?strong>并不是所有事務(wù)隔離級(jí)別都使用間隙鎖,MySQL InnoDB引擎只有在Repeatable Read(默認(rèn))隔離級(jí)別才使用間隙鎖。
間隙鎖的作用只是用來阻止其他事務(wù)在間隙中插入數(shù)據(jù),他不會(huì)阻止其他事務(wù)擁有同樣的的間隙鎖。這就意味著,除了insert語(yǔ)句,允許其他SQL語(yǔ)句可以對(duì)同樣的行加間隙鎖而不會(huì)被阻塞。
對(duì)于唯一索引的加鎖行為,間隙鎖就會(huì)失效,此時(shí)只有記錄鎖起作用。
2. 加鎖語(yǔ)句
前面我們已經(jīng)介紹了InnoDB的是在SQL語(yǔ)句的執(zhí)行過程中通過掃描索引記錄的方式來實(shí)現(xiàn)加鎖行為的。那哪些些語(yǔ)句會(huì)加鎖?加什么樣的鎖?接下來我們逐一描述:
select ... from語(yǔ)句:InnoDB引擎采用多版本并發(fā)控制(MVCC)的方式實(shí)現(xiàn)了非阻塞讀,所以對(duì)于普通的select讀語(yǔ)句,InnoDB并不會(huì)加鎖【注1】。
select ... from lock in share mode語(yǔ)句:這條語(yǔ)句和普通select語(yǔ)句的區(qū)別就是后面加了lock in share mode,通過字面意思我們可以猜到這是一條加鎖的讀語(yǔ)句,并且鎖類型為共享鎖(讀鎖)。InnoDB會(huì)對(duì)搜索的所有索引記錄加next-key鎖,但是如果掃描的唯一索引的唯一行,next-key降級(jí)為索引記錄鎖。
select ... from for update語(yǔ)句:和上面的語(yǔ)句一樣,這條語(yǔ)句加的是排他鎖(寫鎖)。InnoDB會(huì)對(duì)搜索的所有索引記錄加next-key鎖,但是如果掃描唯一索引的唯一行,next-key降級(jí)為索引記錄鎖。
update ... where ...語(yǔ)句:。InnoDB會(huì)對(duì)搜索的所有索引記錄加next-key鎖,但是如果掃描唯一索引的唯一行,next-key降級(jí)為索引記錄鎖!咀2】
delete ... where ...語(yǔ)句:。InnoDB會(huì)對(duì)搜索的所有索引記錄加next-key鎖,但是如果掃描唯一索引的唯一行,next-key降級(jí)為索引記錄鎖。
insert語(yǔ)句:InnoDB只會(huì)在將要插入的那一行上設(shè)置一個(gè)排他的索引記錄鎖。
最后補(bǔ)充兩點(diǎn):
如果一個(gè)查詢使用了輔助索引并且在索引記錄加上了排他鎖,InnoDB會(huì)在相對(duì)應(yīng)的聚合索引記錄上加鎖。
如果你的SQL語(yǔ)句無法使用索引,這樣MySQL必須掃描整個(gè)表以處理該語(yǔ)句,導(dǎo)致的結(jié)果就是表的每一行都會(huì)被鎖定,并且阻止其他用戶對(duì)該表的所有插入。
SQL語(yǔ)句驗(yàn)證
閑言少敘,接下來我們進(jìn)入本文重點(diǎn)SQL語(yǔ)句驗(yàn)證部分。
1.測(cè)試環(huán)境
數(shù)據(jù)庫(kù):MySQL 5.6.35
事務(wù)隔離級(jí)別:Repeatable read
數(shù)據(jù)庫(kù)訪問終端:mysql client
2.驗(yàn)證場(chǎng)景
2.1 場(chǎng)景一
建表:
CREATE TABLE `user` (
`id` int(11) NOT NULL,
`name` varchar(8) NOT NULL,
PRIMARY KEY (`id`),
KEY `name` (`name`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8;
插入數(shù)據(jù):
INSERT INTO `user` (`id`, `name`) VALUES ('1', 'a');
INSERT INTO `user` (`id`, `name`) VALUES ('3', 'c');
INSERT INTO `user` (`id`, `name`) VALUES ('5', 'e');
INSERT INTO `user` (`id`, `name`) VALUES ('7', 'g');
INSERT INTO `user` (`id`, `name`) VALUES ('9', 'i');
首先我們執(zhí)行SQL語(yǔ)句的模板:
步驟 | client 1 | client 2 |
---|
1 | begin; | -- |
2 | SELECT * FROM user where name='e' for update; | -- |
3 | -- | begin; |
4 | -- | INSERT INTO `user` (`id`, `name`) VALUES (10, #{name}); |
5 | rollback; | -- |
6 | -- | rollback; |
替換步驟5中name的值,觀察結(jié)果:
name的值 | 執(zhí)行結(jié)果 |
---|
a | 不阻塞 |
b | 不阻塞 |
d | 阻塞 |
e | 阻塞 |
f | 阻塞 |
h | 不阻塞 |
i | 不阻塞 |
觀察結(jié)果,我們發(fā)現(xiàn)SQL語(yǔ)句
SELECT * FROM user where name='e' for update
一共鎖住索引name中三行記錄,(c,e]區(qū)間應(yīng)該是next-key鎖而(e,h)區(qū)間是索引記錄e后面的間隙。
接下來我們確定next-key鎖中哪部分是索引記錄鎖哪部分是間隙鎖。
執(zhí)行SQL語(yǔ)句的模板:
步驟 | client 1 | client 2 |
---|
1 | begin; | -- |
2 | SELECT * FROM user where name='e' for update; | -- |
3 | -- | SELECT * FROM user where name=#{name} for update; |
5 | rollback; | -- |
6 | -- | rollback; |
替換步驟5中name的值,觀察結(jié)果:
name的值 | 執(zhí)行結(jié)果 |
---|
d | 不阻塞 |
e | 阻塞 |
f | 不阻塞 |
因?yàn)殚g隙鎖只會(huì)阻止insert語(yǔ)句,所以同樣的索引數(shù)據(jù),insert
語(yǔ)句阻塞而select for update
語(yǔ)句不阻塞的就是間隙鎖,如果兩條語(yǔ)句都阻塞就是索引記錄鎖。
觀察執(zhí)行結(jié)果可知,d和f為間隙鎖,e為索引記錄鎖。
結(jié)論:通過兩條SQL,我們確定了對(duì)于輔助索引name在查詢條件為 where name='e'
時(shí)的加鎖范圍為(c,e],(e,g),其中:
對(duì)SQL語(yǔ)句掃描的索引記錄e加索引記錄鎖[e]。
鎖定了e前面的間隙,c到e之間的數(shù)據(jù)(c,e)加了間隙鎖
前兩個(gè)構(gòu)成了next-key鎖(c,e]。
值得注意的是還鎖定了e后面的間隙(e,g)。
說的這里細(xì)心的讀者可能已經(jīng)發(fā)現(xiàn)我們的測(cè)試數(shù)據(jù)中沒有間隙的邊界數(shù)據(jù)c和g。接下來我們就對(duì)間隙邊界值進(jìn)行測(cè)試。
執(zhí)行SQL語(yǔ)句的模板:
步驟 | client 1 | client 2 |
---|
1 | begin; | -- |
2 | SELECT * FROM user where name='e' for update; | -- |
3 | -- | begin; |
4 | -- | INSERT INTO `user` (`id`, `name`) VALUES (#{id}, #{name}); |
5 | rollback; | -- |
6 | -- | rollback; |
替換步驟5中id,name的值,觀察結(jié)果:
id的值 | name=c | 執(zhí)行結(jié)果 | id的值 | name=g | 執(zhí)行結(jié)果 |
---|
-- | -- | -- | -3 | g | 組塞 |
-- | -- | -- | -2 | g | 阻塞 |
-1 | c | 不阻塞 | -1 | g | 阻塞 |
1 | c | 不阻塞 | 1 | g | 不阻塞 |
2 | c | 不阻塞 | 2 | g | 阻塞 |
3 | c | 不阻塞 | 3 | g | 不阻塞 |
4 | c | 阻塞 | 4 | g | 阻塞 |
5 | c | 阻塞 | 5 | g | 阻塞 |
6 | c | 阻塞 | 6 | g | 阻塞 |
7 | c | 不阻塞 | 7 | g | 不阻塞 |
8 | c | 阻塞 | 8 | g | 不阻塞 |
9 | c | 不阻塞 | 9 | g | 不阻塞 |
10 | c | 阻塞 | 10 | g | 不阻塞 |
11 | c | 阻塞 | - | - | - |
12 | c | 阻塞 | - | - | - |
通過觀察以上執(zhí)行結(jié)果,我們發(fā)現(xiàn),name等于c和e時(shí)insert
語(yǔ)句的結(jié)果隨著id值得不同一會(huì)兒鎖定,一會(huì)兒不鎖定。那一定是id列加了鎖才會(huì)造成這樣的結(jié)果。
如果先不看id=5
這一行數(shù)據(jù)的結(jié)果,我們發(fā)現(xiàn)一個(gè)規(guī)律:
當(dāng)name=c
時(shí),name=c
對(duì)應(yīng)的id=3
的id聚合索引數(shù)據(jù)記錄之后的間隙(3,5),(5,7),(7,9),(9,∞)都被加上了鎖。
當(dāng)name=e
時(shí),name=e
對(duì)應(yīng)的id=7
的id聚合索引數(shù)據(jù)記錄之前的間隙(5,7),(3,5),(1,3),(-∞,1)都被加上了鎖。
我們可用select * from user where id = x for update;
語(yǔ)句判斷出以上間隙上加的鎖都為間隙鎖。
接下來我們解釋一下id=5
的鎖定情況
執(zhí)行SQL語(yǔ)句的模板:
步驟 | client 1 | client 2 |
---|
1 | begin; | -- |
2 | SELECT * FROM user where name='e' for update; | -- |
3 | -- | SELECT * FROM user where id=#{id} for update; |
5 | rollback; | -- |
6 | -- | rollback; |
替換步驟5中id的值,觀察結(jié)果:
id的值 | 執(zhí)行結(jié)果 |
---|
3 | 不阻塞 |
4 | 不阻塞 |
5 | 阻塞 |
6 | 不阻塞 |
7 | 不阻塞 |
通過觀察執(zhí)行結(jié)果可知,id=5
的聚合索引記錄上添加了索引記錄鎖。根據(jù)MySQL官方文檔描述,InnoDB引擎在對(duì)輔助索引加鎖的時(shí)候,也會(huì)對(duì)輔助索引所在行所對(duì)應(yīng)的聚合索引(主鍵)加鎖。而主鍵是唯一索引,在對(duì)唯一索引加鎖時(shí),間隙鎖失效,只使用索引記錄鎖。所以SELECT * FROM user where name='e' for update;
不僅對(duì)輔助索引name=e
列加上了next-key鎖,還對(duì)對(duì)應(yīng)的聚合索引id=5
列加上了索引記錄鎖。
最終結(jié)論:
對(duì)于SELECT * FROM user where name='e' for update;
一共有三種鎖定行為:
對(duì)SQL語(yǔ)句掃描過的輔助索引記錄行加上next-key鎖(注意也鎖住記錄行之后的間隙)。
對(duì)輔助索引對(duì)應(yīng)的聚合索引加上索引記錄鎖。
當(dāng)輔助索引為間隙鎖“最小”和“最大”值時(shí),對(duì)聚合索引相應(yīng)的行加間隙鎖!白钚 辨i定對(duì)應(yīng)聚合索引之后的行間隙!白畲蟆敝垫i定對(duì)應(yīng)聚合索引之前的行間隙。
上面我們將對(duì)輔助索引加鎖的情況介紹完了,接下來我們測(cè)試一下對(duì)聚合索引和唯一索引加鎖。
2.2 場(chǎng)景二
建表:
CREATE TABLE `user` (
`id` int(11) NOT NULL,
`name` varchar(8) NOT NULL,
PRIMARY KEY (`id`),
UNIQUE KEY `index_name` (`name`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8;
注意與場(chǎng)景一表user不同的是name列為唯一索引。
插入數(shù)據(jù):
INSERT INTO `user` (`id`, `name`) VALUES ('1', 'a');
INSERT INTO `user` (`id`, `name`) VALUES ('3', 'c');
INSERT INTO `user` (`id`, `name`) VALUES ('5', 'e');
INSERT INTO `user` (`id`, `name`) VALUES ('7', 'g');
INSERT INTO `user` (`id`, `name`) VALUES ('9', 'i');
首先我們執(zhí)行SQL語(yǔ)句的模板:
步驟 | client 1 | client 2 |
---|
1 | begin; | -- |
2 | SELECT * FROM user where name='e' for update; |
3 | -- | begin; |
4 | -- | INSERT INTO `user` (`id`, `name`) VALUES (10, #{name}); |
5 | rollback; | -- |
6 | -- | rollback; |
替換步驟5中name的值,觀察結(jié)果:
name的值 | 執(zhí)行結(jié)果 |
---|
a | 不阻塞 |
b | 不阻塞 |
c | 不阻塞 |
d | 不阻塞 |
e | 阻塞 |
f | 不阻塞 |
g | 不阻塞 |
h | 不阻塞 |
i | 不阻塞 |
由測(cè)試結(jié)果可知,只有name='e'
這行數(shù)據(jù)被鎖定。
通過SQL語(yǔ)句我們驗(yàn)證了,對(duì)于唯一索引列加鎖,間隙鎖失效,
2.3 場(chǎng)景三
場(chǎng)景一和場(chǎng)景二都是在查詢條件等于的情況下做出的范圍判斷,現(xiàn)在我們嘗試一下其他查詢條件,看看結(jié)論是否一致。
借用場(chǎng)景一的表和數(shù)據(jù)。
建表:
CREATE TABLE `user` (
`id` int(11) NOT NULL,
`name` varchar(8) NOT NULL,
PRIMARY KEY (`id`),
KEY `index_name` (`name`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8;
插入數(shù)據(jù):
INSERT INTO `user` (`id`, `name`) VALUES ('1', 'a');
INSERT INTO `user` (`id`, `name`) VALUES ('3', 'c');
INSERT INTO `user` (`id`, `name`) VALUES ('5', 'e');
INSERT INTO `user` (`id`, `name`) VALUES ('7', 'g');
INSERT INTO `user` (`id`, `name`) VALUES ('9', 'i');
執(zhí)行SQL語(yǔ)句的模板:
步驟 | client 1 | client 2 |
---|
1 | begin; | -- |
2 | SELECT * FROM user where name>'e' for update; | -- |
3 | -- | begin; |
4 | -- | INSERT INTO `user` (`id`, `name`) VALUES ('10', #{name}); |
5 | rollback; | -- |
6 | -- | rollback; |
替換步驟5中name的值,觀察結(jié)果:
name的值 | 執(zhí)行結(jié)果 |
---|
a | 阻塞 |
b | 阻塞 |
c | 阻塞 |
d | 阻塞 |
e | 阻塞 |
f | 阻塞 |
g | 阻塞 |
h | 阻塞 |
i | 阻塞 |
這個(gè)結(jié)果是不是和你想象的不太一樣,這個(gè)結(jié)果表明where name>'e'
這個(gè)查詢條件并不是鎖住'e'
列之后的數(shù)據(jù),而鎖住了所有name
列中所有數(shù)據(jù)和間隙。這是為什么呢?
我們執(zhí)行以下的SQL語(yǔ)句執(zhí)行計(jì)劃:
explain select * from user where name>'e' for update;
執(zhí)行結(jié)果:
+----+-------------+-------+-------+---------------+------------+---------+------+------+--------------------------+
id select_type table type possible_keys key key_len ref rows Extra
+----+-------------+-------+-------+---------------+------------+---------+------+------+--------------------------+
1 SIMPLE user index index_name index_name 26 NULL 5 Using where; Using index
+----+-------------+-------+-------+---------------+------------+---------+------+------+--------------------------+
1 row in set (0.00 sec)
如果你的結(jié)果與上面不同先執(zhí)行一下OPTIMIZE TABLE user;
再執(zhí)行以上語(yǔ)句。
通過觀察SQL語(yǔ)句的執(zhí)行計(jì)劃我們發(fā)現(xiàn),語(yǔ)句使用了name
列索引,且rows
參數(shù)等于5,user表中一共也只有5行數(shù)據(jù)。SQL語(yǔ)句的執(zhí)行過程中一共掃描了name
索引記錄5行數(shù)據(jù)且對(duì)這5行數(shù)據(jù)都加上了next-key鎖,符合我們上面的執(zhí)行結(jié)果。
接下來我們?cè)僦圃煲唤M數(shù)據(jù)。
建表:
CREATE TABLE `user` (
`id` int(11) NOT NULL,
`name` varchar(8) NOT NULL,
`age` int(11) NOT NULL,
PRIMARY KEY (`id`),
KEY `index_name` (`name`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8;
插入數(shù)據(jù):
INSERT INTO `user` (`id`, `name`,`age`) VALUES ('1', 'a','15');
INSERT INTO `user` (`id`, `name`,`age`) VALUES ('3', 'c','20');
INSERT INTO `user` (`id`, `name`,`age`) VALUES ('5', 'e','16');
INSERT INTO `user` (`id`, `name`,`age`) VALUES ('7', 'g','19');
INSERT INTO `user` (`id`, `name`,`age`) VALUES ('9', 'i','34');
這張表和前表的區(qū)別是多了一列非索引列age
。
我們?cè)賵?zhí)行一下同樣的SQL語(yǔ)句執(zhí)行計(jì)劃:
explain select * from user where name>'e' for update;
執(zhí)行結(jié)果:
+----+-------------+-------+-------+---------------+------------+---------+------+------+-----------------------+
id select_type table type possible_keys key key_len ref rows Extra
+----+-------------+-------+-------+---------------+------------+---------+------+------+-----------------------+
1 SIMPLE user range index_name index_name 26 NULL 2 Using index condition
+----+-------------+-------+-------+---------------+------------+---------+------+------+-----------------------+
1 row in set (0.00 sec)
是不是和第一次執(zhí)行結(jié)果不同了,rows
參數(shù)等于2,說明掃描了兩行記錄,結(jié)合SQL語(yǔ)句select * from user where name>'e' for update;
執(zhí)行后返回結(jié)果我們判斷這兩行記錄應(yīng)該為g和i。
因?yàn)?code>select * from user where name>'e' for update;語(yǔ)句掃描了兩行索引記錄分別是g和i,所以我們將g和i的鎖定范圍疊就可以得到where name>'e'
的鎖定范圍:
索引記錄g在name
列鎖定范圍為(e,g],(g,i)。索引記錄i的在name
列鎖定范圍為(g,i],(i,+∞)。兩者疊加后鎖定范圍為(e,g],(g,i],(i,+∞)。其中g(shù),i為索引記錄鎖。
g和i對(duì)應(yīng)id
列中的7和9加索引記錄鎖。
當(dāng)name
列的值為鎖定范圍上邊界e時(shí),還會(huì)在e所對(duì)應(yīng)的id
列值為5之后的所有值之間加上間隙鎖,范圍為(5,7),(7,9),(9,+∞)。下邊界為+∞無需考慮。
接下來我們逐一測(cè)試:
首先測(cè)試驗(yàn)證了next-key鎖范圍,執(zhí)行SQL語(yǔ)句的模板:
步驟 | client 1 | client 2 |
---|
1 | begin; | -- |
2 | SELECT * FROM user where name>'e' for update; | -- |
3 | -- | begin; |
4 | -- | INSERT INTO `user` (`id`, `name`, `age`) VALUES ('10', #{name},'18'); |
5 | rollback; | -- |
6 | -- | rollback; |
替換步驟5中name的值,觀察結(jié)果:
name的值 | 執(zhí)行結(jié)果 |
---|
a | 不阻塞 |
b | 不阻塞 |
c | 不阻塞 |
d | 不阻塞 |
f | 阻塞 |
g | 阻塞 |
h | 阻塞 |
i | 阻塞 |
j | 阻塞 |
k | 阻塞 |
下面驗(yàn)證next-key鎖中哪部分是間隙鎖,哪部分是索引記錄鎖,執(zhí)行SQL語(yǔ)句的模板:
步驟 | client 1 | client 2 |
---|
1 | begin; | -- |
2 | SELECT * FROM user where name>'e' for update; | -- |
3 | -- | SELECT * FROM user where name=#{name} for update; |
5 | rollback; | -- |
6 | -- | rollback; |
替換步驟5中name的值,觀察結(jié)果:
name的值 | 執(zhí)行結(jié)果 |
---|
e | 不阻塞 |
f | 不阻塞 |
g | 阻塞 |
h | 不阻塞 |
i | 阻塞 |
j | 不阻塞 |
接下來驗(yàn)證對(duì)id
列加索引記錄鎖,執(zhí)行SQL語(yǔ)句的模板:
步驟 | client 1 | client 2 |
---|
1 | begin; | -- |
2 | SELECT * FROM user where name>'e' for update; | -- |
3 | -- | SELECT * FROM user where id=#{id} for update; |
5 | rollback; | -- |
6 | -- | rollback; |
替換步驟5中id的值,觀察結(jié)果:
id的值 | 執(zhí)行結(jié)果 |
---|
5 | 不阻塞 |
6 | 不阻塞 |
7 | 阻塞 |
8 | 不阻塞 |
9 | 阻塞 |
10 | 不阻塞 |
最后我們驗(yàn)證name
列的值為邊界數(shù)據(jù)e時(shí),id
列間隙鎖的范圍,執(zhí)行SQL語(yǔ)句的模板:
步驟 | client 1 | client 2 |
---|
1 | begin; | -- |
2 | SELECT * FROM user where name>'e' for update; | -- |
3 | -- | begin; |
4 | -- | INSERT INTO `user` (`id`, `name`,`age`) VALUES (#{id}, 'e','18'); |
5 | rollback; | -- |
6 | -- | rollback; |
替換步驟5中id的值,觀察結(jié)果:
id的值 | 執(zhí)行結(jié)果 |
---|
-1 | 不阻塞 |
1 | 不阻塞 |
2 | 不阻塞 |
3 | 不阻塞 |
4 | 不阻塞 |
5 | 不阻塞 |
6 | 阻塞 |
7 | 阻塞 |
8 | 阻塞 |
9 | 阻塞 |
10 | 阻塞 |
11 | 阻塞 |
12 | 阻塞 |
注意7和9是索引記錄鎖記錄鎖。
觀察上面的所有SQL語(yǔ)句執(zhí)行結(jié)果,可以驗(yàn)證select * from user where name>'e' for update
的鎖定范圍為此語(yǔ)句掃描name
列索引記錄g和i的鎖定范圍的疊加組合。
2.4 場(chǎng)景四
我們通過場(chǎng)景三驗(yàn)證了普通索引的范圍查詢語(yǔ)句加鎖范圍,現(xiàn)在我們來驗(yàn)證一下唯一索引的范圍查詢情況下的加鎖范圍。有了場(chǎng)景三的鋪墊我們直接跳過掃描全部索引的情況,創(chuàng)建可以掃描范圍記錄的表結(jié)構(gòu)并插入相應(yīng)數(shù)據(jù)測(cè)試。
建表:
CREATE TABLE `user` (
`id` int(11) NOT NULL,
`name` varchar(8) NOT NULL,
`age` int(11) NOT NULL,
PRIMARY KEY (`id`),
UNIQUE KEY `index_name` (`name`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8;
插入數(shù)據(jù):
INSERT INTO `user` (`id`, `name`,`age`) VALUES ('1', 'a','15');
INSERT INTO `user` (`id`, `name`,`age`) VALUES ('3', 'c','20');
INSERT INTO `user` (`id`, `name`,`age`) VALUES ('5', 'e','16');
INSERT INTO `user` (`id`, `name`,`age`) VALUES ('7', 'g','19');
INSERT INTO `user` (`id`, `name`,`age`) VALUES ('9', 'i','34');
和場(chǎng)景三表唯一不同是name
列為唯一索引。
SQL語(yǔ)句select * from user where name>'e'
掃描name
列兩條索引記錄g和i。如果需要只對(duì)g和i這兩條記錄加上記錄鎖無法避免幻讀的發(fā)生,索引鎖定范圍應(yīng)該還是兩條數(shù)據(jù)next-key鎖鎖的組合:(e,g],(g,i],(i,+∞)。其中g(shù),i為索引記錄鎖。
我們通過SQL驗(yàn)證我們的結(jié)論,執(zhí)行SQL語(yǔ)句的模板:
步驟 | client 1 | client 2 |
---|
1 | begin; | -- |
2 | SELECT * FROM user where name>'e' for update; | -- |
3 | -- | begin; |
4 | -- | INSERT INTO `user` (`id`, `name`, `age`) VALUES ('10', #{name},'18'); |
5 | rollback; | -- |
6 | -- | rollback; |
替換步驟5中name的值,觀察結(jié)果:
name的值 | 執(zhí)行結(jié)果 |
---|
a | 不阻塞 |
b | 不阻塞 |
c | 不阻塞 |
d | 不阻塞 |
f | 阻塞 |
g | 阻塞 |
h | 阻塞 |
i | 阻塞 |
j | 阻塞 |
k | 阻塞 |
下面驗(yàn)證next-key鎖中哪部分是間隙鎖,哪部分是索引記錄鎖,執(zhí)行SQL語(yǔ)句的模板:
步驟 | client 1 | client 2 |
---|
1 | begin; | -- |
2 | SELECT * FROM user where name>'e' for update; | -- |
3 | -- | SELECT * FROM user where name=#{name} for update; |
5 | rollback; | -- |
6 | -- | rollback; |
替換步驟5中name的值,觀察結(jié)果:
name的值 | 執(zhí)行結(jié)果 |
---|
e | 不阻塞 |
f | 不阻塞 |
g | 阻塞 |
h | 不阻塞 |
i | 阻塞 |
j | 不阻塞 |
通過上面兩條SQL語(yǔ)句的驗(yàn)證結(jié)果,我們證明了我們的g和i的鎖定范圍趨勢(shì)為兩者next-key疊加組合。
接下來我們驗(yàn)證一下對(duì)輔助索引加鎖后對(duì)聚合索引的鎖轉(zhuǎn)移,執(zhí)行SQL語(yǔ)句的模板:
步驟 | client 1 | client 2 |
---|
1 | begin; | -- |
2 | SELECT * FROM user where name>'e' for update; | -- |
3 | -- | SELECT * FROM user where id=#{id} for update; |
5 | rollback; | -- |
6 | -- | rollback; |
替換步驟5中id的值,觀察結(jié)果:
id的值 | 執(zhí)行結(jié)果 |
---|
5 | 不阻塞 |
6 | 不阻塞 |
7 | 阻塞 |
8 | 不阻塞 |
9 | 阻塞 |
10 | 不阻塞 |
由結(jié)果可知對(duì)輔助索引name
中的g和i列對(duì)應(yīng)的聚合索引id
列中的7和9加上了索引記錄鎖。
到目前為止所有實(shí)驗(yàn)結(jié)果和場(chǎng)景三完全一樣,這也很好理解,畢竟場(chǎng)景四和場(chǎng)景三只是輔助索引name
的索引類型不同,一個(gè)是唯一索引,一個(gè)是普通索引。
最后驗(yàn)證意向,next-key鎖邊界數(shù)據(jù)e,看看結(jié)論時(shí)候和場(chǎng)景三相同。
執(zhí)行SQL語(yǔ)句的模板:
步驟 | client 1 | client 2 |
---|
1 | begin; | -- |
2 | SELECT * FROM user where name>'e' for update; | -- |
3 | -- | begin; |
4 | -- | INSERT INTO `user` (`id`, `name`,`age`) VALUES (#{id}, 'e','18'); |
5 | rollback; | -- |
6 | -- | rollback; |
替換步驟5中id的值,觀察結(jié)果:
id的值 | 執(zhí)行結(jié)果 |
---|
-1 | 不阻塞 |
1 | 不阻塞 |
2 | 不阻塞 |
3 | 不阻塞 |
4 | 不阻塞 |
5 | 不阻塞 |
6 | 不阻塞 |
7 | 阻塞 |
8 | 不阻塞 |
9 | 阻塞 |
10 | 不阻塞 |
11 | 不阻塞 |
12 | 不阻塞 |
注意7和9是索引記錄鎖記錄鎖。
通過結(jié)果可知,當(dāng)name
列為索引記錄上邊界e時(shí),并沒有對(duì)id有加鎖行為,這點(diǎn)與場(chǎng)景三不同。
對(duì)于唯一索引的范圍查詢和普通索引的范圍查詢類似,唯一不同的是當(dāng)輔助索引等于上下范圍的邊界值是不會(huì)對(duì)主鍵加上間隙鎖。
唯一索引范圍查詢加鎖范圍:
結(jié)論
InnoDB引擎會(huì)對(duì)他掃描過的索引記錄加上相應(yīng)的鎖,通過“場(chǎng)景一”我們已經(jīng)明確了掃描一條普通索引記錄的鎖定范圍,通過“場(chǎng)景三”我們可以推斷任意多個(gè)掃描普通索引索引記錄的鎖定范圍。通過“場(chǎng)景二”我們確定了掃描一條唯一索引記錄(或主鍵)的鎖定范圍。通過“場(chǎng)景四”我們可以推斷任意多個(gè)掃描索唯一引記錄(或主鍵)的鎖定范圍。在實(shí)際的應(yīng)用可以靈活使用,判斷兩條SQL語(yǔ)句是否相互鎖定。這里還需要注意的是對(duì)于索引的查詢條件,不能想當(dāng)然的理解,他往往不是我們理解的樣子,需要結(jié)合執(zhí)行計(jì)劃判斷索引最終掃描的記錄數(shù),否則會(huì)對(duì)加鎖范圍理解產(chǎn)生偏差。
備注
注1:在事務(wù)隔離級(jí)別為SERIALIZABLE時(shí),普通的select語(yǔ)句也會(huì)對(duì)語(yǔ)句執(zhí)行過程中掃描過的索引加上next-key鎖。如果語(yǔ)句掃描的是唯一索引,那就將next-key鎖降級(jí)為索引記錄鎖了。
注2:當(dāng)更新語(yǔ)句修改聚合索引(主鍵)記錄時(shí),會(huì)對(duì)受影響的輔助索引執(zhí)行隱性的加鎖操作。當(dāng)插入新的輔助索引記錄之前執(zhí)行重復(fù)檢查掃描時(shí)和當(dāng)插入新的輔助索引記錄時(shí),更新操作還對(duì)受影響的輔助索引記錄添加共享鎖。
相關(guān)推薦:
mysql執(zhí)行sql文件報(bào)錯(cuò)Error: Unknown storage engine‘InnoDB如何解決
MySQL啟動(dòng)時(shí)InnoDB引擎被禁用了怎么辦
MySQL存儲(chǔ)引擎MyISAM和InnoDB之間的比較
以上就是MySQL數(shù)據(jù)庫(kù)InnoDB引擎行級(jí)鎖鎖定范圍詳解的詳細(xì)內(nèi)容,更多請(qǐng)關(guān)注php中文網(wǎng)其它相關(guān)文章!
學(xué)習(xí)教程快速掌握從入門到精通的SQL知識(shí)。