您好,登錄后才能下訂單哦!
本篇內容主要講解“分析MySQL數據庫Innodb中的事務隔離級別和鎖的關系”,感興趣的朋友不妨來看看。本文介紹的方法操作簡單快捷,實用性強。下面就讓小編來帶大家學習“分析MySQL數據庫Innodb中的事務隔離級別和鎖的關系”吧!
update class_teacher set class_name='初三三班' where id=1;
commit; select id,class_name,teacher_id from class_teacher where teacher_id=1;
idclass_nameteacher_id1初三三班12初三一班1讀到了事務B修改的數據,和第一次查詢的結果不一樣,是不可重讀的。
commit;
事務B修改id=1的數據提交之后,事務A同樣的查詢,后一次和前一次的結果不一樣,這就是不可重讀(重新讀取產生的結果不一樣)。這就很可能帶來一些問題,那么我們來看看在RR級別中MySQL的表現:
事務A | 事務B | 事務C |
---|---|---|
begin; | begin; | begin; |
select id,class_name,teacher_id from class_teacher where teacher_id=1; idclass_nameteacher_id1初三二班12初三一班1 | ||
update class_teacher set class_name='初三三班' where id=1; commit;
| ||
insert into class_teacher values (null,'初三三班',1); commit; | ||
select id,class_name,teacher_id from class_teacher where teacher_id=1; idclass_nameteacher_id1初三二班12初三一班1沒有讀到事務B修改的數據,和第一次sql讀取的一樣,是可重復讀的。 沒有讀到事務C新添加的數據。 | ||
commit; |
我們注意到,當teacher_id=1時,事務A先做了一次讀取,事務B中間修改了id=1的數據,并commit之后,事務A第二次讀到的數據和第一次完全相同。所以說它是可重讀的。那么MySQL是怎么做到的呢?這里姑且賣個關子,我們往下看。
####不可重復讀和幻讀的區別####
很多人容易搞混不可重復讀和幻讀,確實這兩者有些相似。但不可重復讀重點在于update和delete,而幻讀的重點在于insert。
如果使用鎖機制來實現這兩種隔離級別,在可重復讀中,該sql第一次讀取到數據后,就將這些數據加鎖,其它事務無法修改這些數據,就可以實現可重復讀了。但這種方法卻無法鎖住insert的數據,所以當事務A先前讀取了數據,或者修改了全部數據,事務B還是可以insert數據提交,這時事務A就會發現莫名其妙多了一條之前沒有的數據,這就是幻讀,不能通過行鎖來避免。需要Serializable隔離級別 ,讀用讀鎖,寫用寫鎖,讀鎖和寫鎖互斥,這么做可以有效的避免幻讀、不可重復讀、臟讀等問題,但會極大的降低數據庫的并發能力。
所以說不可重復讀和幻讀最大的區別,就在于如何通過鎖機制來解決他們產生的問題。
上文說的,是使用悲觀鎖機制來處理這兩種問題,但是MySQL、ORACLE、PostgreSQL等成熟的數據庫,出于性能考慮,都是使用了以樂觀鎖為理論基礎的MVCC(多版本并發控制)來避免這兩種問題。
####悲觀鎖和樂觀鎖####
悲觀鎖
正如其名,它指的是對數據被外界(包括本系統當前的其他事務,以及來自外部系統的事務處理)修改持保守態度,因此,在整個數據處理過程中,將數據處于鎖定狀態。悲觀鎖的實現,往往依靠數據庫提供的鎖機制(也只有數據庫層提供的鎖機制才能真正保證數據訪問的排他性,否則,即使在本系統中實現了加鎖機制,也無法保證外部系統不會修改數據)。
在悲觀鎖的情況下,為了保證事務的隔離性,就需要一致性鎖定讀。讀取數據時給加鎖,其它事務無法修改這些數據。修改刪除數據時也要加鎖,其它事務無法讀取這些數據。
樂觀鎖
相對悲觀鎖而言,樂觀鎖機制采取了更加寬松的加鎖機制。悲觀鎖大多數情況下依靠數據庫的鎖機制實現,以保證操作最大程度的獨占性。但隨之而來的就是數據庫性能的大量開銷,特別是對長事務而言,這樣的開銷往往無法承受。
而樂觀鎖機制在一定程度上解決了這個問題。樂觀鎖,大多是基于數據版本( Version )記錄機制實現。何謂數據版本?即為數據增加一個版本標識,在基于數據庫表的版本解決方案中,一般是通過為數據庫表增加一個 “version” 字段來實現。讀取出數據時,將此版本號一同讀出,之后更新時,對此版本號加一。此時,將提交數據的版本數據與數據庫表對應記錄的當前版本信息進行比對,如果提交的數據版本號大于數據庫表當前版本號,則予以更新,否則認為是過期數據。
要說明的是,MVCC的實現沒有固定的規范,每個數據庫都會有不同的實現方式,這里討論的是InnoDB的MVCC。
####MVCC在MySQL的InnoDB中的實現
在InnoDB中,會在每行數據后添加兩個額外的隱藏的值來實現MVCC,這兩個值一個記錄這行數據何時被創建,另外一個記錄這行數據何時過期(或者被刪除)。 在實際操作中,存儲的并不是時間,而是事務的版本號,每開啟一個新事務,事務的版本號就會遞增。 在可重讀Repeatable reads事務隔離級別下:
SELECT時,讀取創建版本號<=當前事務版本號,刪除版本號為空或>當前事務版本號。
INSERT時,保存當前事務版本號為行的創建版本號
DELETE時,保存當前事務版本號為行的刪除版本號
UPDATE時,插入一條新紀錄,保存當前事務版本號為行創建版本號,同時保存當前事務版本號到原來刪除的行
通過MVCC,雖然每行記錄都需要額外的存儲空間,更多的行檢查工作以及一些額外的維護工作,但可以減少鎖的使用,大多數讀操作都不用加鎖,讀數據操作很簡單,性能很好,并且也能保證只會讀取到符合標準的行,也只鎖住必要行。
我們不管從數據庫方面的教課書中學到,還是從網絡上看到,大都是上文中事務的四種隔離級別這一模塊列出的意思,RR級別是可重復讀的,但無法解決幻讀,而只有在Serializable級別才能解決幻讀。于是我就加了一個事務C來展示效果。在事務C中添加了一條teacher_id=1的數據commit,RR級別中應該會有幻讀現象,事務A在查詢teacher_id=1的數據時會讀到事務C新加的數據。但是測試后發現,在MySQL中是不存在這種情況的,在事務C提交后,事務A還是不會讀到這條數據。可見在MySQL的RR級別中,是解決了幻讀的讀問題的。參見下圖
讀問題解決了,根據MVCC的定義,并發提交數據時會出現沖突,那么沖突時如何解決呢?我們再來看看InnoDB中RR級別對于寫數據的處理。
####“讀”與“讀”的區別
可能有讀者會疑惑,事務的隔離級別其實都是對于讀數據的定義,但到了這里,就被拆成了讀和寫兩個模塊來講解。這主要是因為MySQL中的讀,和事務隔離級別中的讀,是不一樣的。
我們且看,在RR級別中,通過MVCC機制,雖然讓數據變得可重復讀,但我們讀到的數據可能是歷史數據,是不及時的數據,不是數據庫當前的數據!這在一些對于數據的時效特別敏感的業務中,就很可能出問題。
對于這種讀取歷史數據的方式,我們叫它快照讀 (snapshot read),而讀取數據庫當前版本數據的方式,叫當前讀 (current read)。很顯然,在MVCC中:
快照讀:就是select
select * from table ....;
當前讀:特殊的讀操作,插入/更新/刪除操作,屬于當前讀,處理的都是當前的數據,需要加鎖。
select * from table where ? lock in share mode;
select * from table where ? for update;
insert;
update ;
delete;
事務的隔離級別實際上都是定義了當前讀的級別,MySQL為了減少鎖處理(包括等待其它鎖)的時間,提升并發能力,引入了快照讀的概念,使得select不用加鎖。而update、insert這些“當前讀”,就需要另外的模塊來解決了。
###寫("當前讀")
事務的隔離級別中雖然只定義了讀數據的要求,實際上這也可以說是寫數據的要求。上文的“讀”,實際是講的快照讀;而這里說的“寫”就是當前讀了。
為了解決當前讀中的幻讀問題,MySQL事務使用了Next-Key鎖。
####Next-Key鎖
Next-Key鎖是行鎖和GAP(間隙鎖)的合并,行鎖上文已經介紹了,接下來說下GAP間隙鎖。
行鎖可以防止不同事務版本的數據修改提交時造成數據沖突的情況。但如何避免別的事務插入數據就成了問題。我們可以看看RR級別和RC級別的對比
RC級別:
事務A | 事務B |
---|---|
begin; | begin; |
select id,class_name,teacher_id from class_teacher where teacher_id=30; idclass_nameteacher_id2初三二班30
| |
update class_teacher set class_name='初三四班' where teacher_id=30; | |
insert into class_teacher values (null,'初三二班',30); commit; | |
select id,class_name,teacher_id from class_teacher where teacher_id=30; idclass_nameteacher_id2初三四班3010初三二班30
|
RR級別:
事務A | 事務B |
---|---|
begin; | begin; |
select id,class_name,teacher_id from class_teacher where teacher_id=30; idclass_nameteacher_id2初三二班30 | |
update class_teacher set class_name='初三四班' where teacher_id=30; | |
insert into class_teacher values (null,'初三二班',30); waiting.... | |
select id,class_name,teacher_id from class_teacher where teacher_id=30; idclass_nameteacher_id2初三四班30 | |
commit; | 事務Acommit后,事務B的insert執行。 |
通過對比我們可以發現,在RC級別中,事務A修改了所有teacher_id=30的數據,但是當事務Binsert進新數據后,事務A發現莫名其妙多了一行teacher_id=30的數據,而且沒有被之前的update語句所修改,這就是“當前讀”的幻讀。
RR級別中,事務A在update后加鎖,事務B無法插入新數據,這樣事務A在update前后讀的數據保持一致,避免了幻讀。這個鎖,就是Gap鎖。
MySQL是這么實現的:
在class_teacher這張表中,teacher_id是個索引,那么它就會維護一套B+樹的數據關系,為了簡化,我們用鏈表結構來表達(實際上是個樹形結構,但原理相同)
如圖所示,InnoDB使用的是聚集索引,teacher_id身為二級索引,就要維護一個索引字段和主鍵id的樹狀結構(這里用鏈表形式表現),并保持順序排列。
Innodb將這段數據分成幾個個區間
(negative infinity, 5],
(5,30],
(30,positive infinity);
update class_teacher set class_name='初三四班' where teacher_id=30;不僅用行鎖,鎖住了相應的數據行;同時也在兩邊的區間,(5,30]和(30,positive infinity),都加入了gap鎖。這樣事務B就無法在這個兩個區間insert進新數據。
受限于這種實現方式,Innodb很多時候會鎖住不需要鎖的區間。如下所示:
事務A | 事務B | 事務C |
---|---|---|
begin; | begin; | begin; |
select id,class_name,teacher_id from class_teacher; idclass_nameteacher_id1初三一班52初三二班30 | ||
update class_teacher set class_name='初一一班' where teacher_id=20; | ||
insert into class_teacher values (null,'初三五班',10); waiting ..... | insert into class_teacher values (null,'初三五班',40); | |
commit; | 事務A commit之后,這條語句才插入成功 | commit; |
commit; |
update的teacher_id=20是在(5,30]區間,即使沒有修改任何數據,Innodb也會在這個區間加gap鎖,而其它區間不會影響,事務C正常插入。
如果使用的是沒有索引的字段,比如update class_teacher set teacher_id=7 where class_name='初三八班(即使沒有匹配到任何數據)',那么會給全表加入gap鎖。同時,它不能像上文中行鎖一樣經過MySQL Server過濾自動解除不滿足條件的鎖,因為沒有索引,則這些字段也就沒有排序,也就沒有區間。除非該事務提交,否則其它事務無法插入任何數據。
行鎖防止別的事務修改或刪除,GAP鎖防止別的事務新增,行鎖和GAP鎖結合形成的的Next-Key鎖共同解決了RR級別在寫數據時的幻讀問題。
###Serializable
這個級別很簡單,讀加共享鎖,寫加排他鎖,讀寫互斥。使用的悲觀鎖的理論,實現簡單,數據更加安全,但是并發能力非常差。如果你的業務并發的特別少或者沒有并發,同時又要求數據及時可靠的話,可以使用這種模式。
這里要吐槽一句,不要看到select就說不會加鎖了,在Serializable這個級別,還是會加鎖的!
到此,相信大家對“分析MySQL數據庫Innodb中的事務隔離級別和鎖的關系”有了更深的了解,不妨來實際操作一番吧!這里是億速云網站,更多相關內容可以進入相關頻道進行查詢,關注我們,繼續學習!
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。