Mysql InnoDB多版本並發控制MVCC

參考書籍《mysql是怎樣運行的》

系列文章目錄和關於我

一丶為什麼需要事務隔離級別

mysql是一個客戶端/服務斷軟件,對於同一個服務器來說,可以有多個客戶端進行連接,每一個客戶端進行連接之後就形成一個會話,每一個客戶端都可以在自己的會話中向服務器發出請求語句,一個請求語句可能是某一個事務的一部分,服務器可以同時處理多個事務。

如果事務時一個接着一個進行,那麼下一個事務是在上一個事務的一致性前提下進行的,就沒用一致性的問題,但是事務是並發進行且可能訪問到相同的數據這時候就會出現如下問題

image-20221110071339372

可以看到AB最開始總和13元,最後AB總和18元,銀行血虧五元,這顯然違背了一致性——錢的總量不變。這就是並發情況下兩個事務的影響,所以需要事務隔離讓事務隔離的進行,互不干涉。

1.實現事務隔離的方式:串行執行

最簡單直接的方式,同一時間只能有一個事務運行,這樣必然不會有上述不一致的情況,但是大大降低了吞吐率並增加了事務的等待時間

2.實現事務隔離的方式:可串行執行

並發事務之所以出現不一致的情況,就是由於多個事務訪問相同的數據,需要實現多個事務在訪問相同數據的時候進行限制,比方說上圖中事務2想訪問A賬戶的值需要等待事務提交事務之後,這樣可以讓並發事務的執行如同串行執行的效果一樣。

二丶並發事務執行的問題:臟寫,臟讀,不可重複讀,幻讀

1.臟寫

一個事務修改了另外一個未提交事務修改過的數據

  • 臟寫導致一致性無法保證

    img

    上圖事務A和事務B都更新紫色數據,其中事務A首先更新為A,然後事務B過來更新為B,這時候事務A回滾後更新為Null,事務 B 明明正常寫了一行數據,但是寫完之後發現值變了,有點丟失更新的意思。(比如A表示餘額,這時候在將餘額A判斷是否足以支付,判斷得到可以,事務B執行扣費寫入A-5,商家收到5元,結果這時候回滾了,A變成Null,事務A中轉錢的一方錢變為A,錢的總額變為A+5了)

  • 臟寫導致原子性受到破壞

    假如上述的事務B還操作了另外的數據,比如插入一條數據C,並且更改為B寫入C是在一個事務下面的,需要具備原子性,但是臟寫讓B的更改需要部分回滾為Null,這樣插入C和更改B就不具備原子性(比如A表示餘額,這時候在將餘額A判斷是否足以支付,判斷得到可以,事務B執行扣費寫入A-5,商家收到5元,結果這時候回滾了,A變成Null,這時候部分回滾,商家的5元沒用回滾,商家的庫存也沒用回滾,原子性被破壞)

2.臟讀

如果一個事務讀取到另外一個事務未提交的數據,意味着發生了臟讀

img

比如事務A先寫數據A,然後事務B督導數據A後在內存中使用A進行一系列操作(比如A表示餘額,這時候在將餘額A判斷是否足以支付,判斷得到可以)但是事務A這時候回滾了,事務B再次讀取數據發現為null,這就是臟讀。

臟讀可能引發一致性的問題:比如事務操作時修改x和y的值,並且二者總是相等的,A修改x為1,還沒來得及修改y也沒用提交事務,這時候事務B讀取x=1,y=0,二者不等,事務B讀取到了數據庫不一致的狀態,讀取到未提交事務的值

3.不可重複讀

假如一個事務修改了另外一個事務未提交的數據,意味發生了不可重複讀

img

比如事務A第一次讀取到值為A,接着事務B修改為B,並且提交了事務B,然後事務A再次讀取得到的數據是B,同一行數據多次讀取值並不相同,這稱作不可重複讀。它是指在同一個事務裏面查詢同一行數據,每次查到的數據都不一樣。和臟讀區別在於臟讀是由於別的事務回滾導致,而不可重複讀讀到的其實是已經提交的數據。

事務A讀到事務B提交後的數據似乎很合理,但是我們想像這樣一種場景:你有一個流水表和用戶餘額,其中記錄用戶每天的流水,你在月初0點的時候核對流水和庫存,但是流水很多,你的程序選擇一個一個用戶的進行核對,核對用戶甲,甲沒做任何消費,但是當你核對B的時候,你將B的流水load到內存中,但是B這時候(0點30分,這一筆數據新的一個餘額)進行了扣除餘額的操作,導致B餘額和流水對不上了。

4.幻讀

如果一個事務A先根據沒用搜索條件查詢到一些記錄,在該事務未提交前,另外一個事務寫入(delete,update,insert)了符合搜索條件的記錄,這時候事務A再次讀取,發現數據條數和第一次讀取的不同,如同出現了幻覺,稱之為幻讀

img

事務A讀到事務B提交後的數據似乎很合理,但是我們想像這樣一種場景:你有一個需求將會公司的男性員工了女性員工查詢進行展示,你先查詢了總數為100人,然後查詢男性的總數50人,後查詢女性人數準備在頁面展示共100人,其中男50人,女50人,結果這是管理信息的人發現有一位員工性別錯誤錄入了,將其從男修改為女,這時候你讀取事務就是女51人了,你在主頁顯示了共100人,其中男50人,女51人

三丶隔離級別

1.Read UnCommitted 讀未提交

在此隔離級別下,會發生臟讀,不可重複讀,和幻讀

2.Read Committed 讀已提交

在此隔離級別下,會發生不可重複讀,和幻讀

3.Repeatable Read 可重複讀

在此隔離級別下,可能發生幻讀

4.Serializable 可串行化

在此隔離級別下,不會發生臟讀,不可重複讀,和幻讀

其中臟寫是對一致性影響最嚴重的,無論是何種隔離級別,都不允許臟寫發生,innodb使用鎖保證不會出現臟寫現象,第一個事務更新某條記錄的時候,會給這條記錄加鎖,另外一個事務在此更新的時候,需要等待第一個事務提交釋放鎖後更新。隔離級別越高,其並發能力越低。

四丶Mysql設置隔離級別

默認隔離級別可重複讀

1.設置全局隔離級別

SET GLOBAL TRANSACTION ISOLATION LEVEL 期望的隔離級別(可選READ UNCOMMITED,READ COMMITED,REPEATABLE READ,SERIALIZABLE),此命令只對執行語句後新產生的會話有效,對當前已經存在的會話無效

2.設置會話隔離級別

SET SESSION TRANSACTION ISOLATION LEVEL 期望的隔離級別(可選READ UNCOMMITED,READ COMMITED,REPEATABLE READ,SERIALIZABLE),對當前會話後續事務有效,該語句可以在已開啟的事務中執行,但是不會影響當前正在執行的事務,如果在事務之間執行,只會對後續的事務有效

3.設置下一個事務的隔離級別

SET TRANSACTION ISOLATION LEVEL 期望的隔離級別(可選READ UNCOMMITED,READ COMMITED,REPEATABLE READ,SERIALIZABLE) 只對當前會話的下一個即將開啟的事務有效,下一個事務執行完後,後續事務將恢復到之前的隔離級別,該語句不能再已經開啟的事務中執行,否則會報錯。

4.指定服務器的隔離級別

在啟動的時候使用--transaction-isolation=xxx即可執行默認隔離級別

五丶MVCC原理

下面討論記錄對當前事務是否可見都是基於當前事務中執行的查詢是快照讀(普通查詢),對於當前讀(select xxx for update,select xxx lock in share mode)是不通用的

1.版本鏈

對於InnoDB存儲引擎來說,其聚簇索引記錄中包含兩個隱藏列:

  • trx_id:一個事務每次對聚簇索引記錄做出改動的時候,都會把該事務的事務id複製給此列
  • roll_point:每次對某條聚簇索引記錄進行改動的時,都會把舊的版本寫入到undo 日誌中,此列相當於一個指針,指向修改前的信息

image-20221113141251522

每次修改都會形成Undo 日誌,所有版本的數據會通過roll_point串聯成一個鏈表,稱之為版本鏈,頭節點是當前記錄的最新值。利用版本鏈控制多個並發事務訪問相同記錄時的行為稱為MVCC多版本並發控制。

其實在undo日誌中,只記錄被更新列的信息,而不是記錄全部的信息,對於沒有記錄的列,會通過版本鏈找少一個版本中的對應列的信息,直到找到聚簇索引葉子節點中的內容

2.Read View

對於使用Read Uncommitted隔離級別的事務,可以讀取到沒提交的數據,那麼直接讀取最新的版本即可。對於Serializable隔離級別,innodb直接通過加鎖來訪問記錄。對於read committed 和 repeatable read隔離級別的事務,都必須保證督導的數據是已經提交事務修改過的記錄,那麼如何判斷版本鏈中的哪個版本的數據是當前事務可見的昵?

innodb 使用的Read View

2.1 read view 的結構

image-20221113143118836

  • m_ids:在生成read view時,當前系統中活躍的讀寫事務id列表
  • min_trx_id:生成read view時,當前系統中活躍的讀寫事務中最西澳的事務id,也就是m_ids中的最小值
  • max_trx_id:生成read view時,系統應該分配給下一個事務的事務id值
  • creator_trx_id:生成該read view的事務的事務id

2.2 read view 判斷某個版本當前事務釋放可見的步驟

  1. 如果被訪問版本的trx_idcreator_trx_id相同,意味着當前事務在訪問自己修改的記錄,自然可見
  2. 如果訪問版本的trx_id屬性值小於read view中的min_trx_id 表明此版本是生成read view之前已經提交的事務,那麼自然可見
  3. 如果訪問版本的trx_id,大於等於read view中的max_trx_id說明,當前版本數據是生成read view後開啟事務產生的,那麼自然不可見
  4. 如果訪問版本的trx_id 介於min_trx_idmax_trx_id之間,需要判斷trx_id是否位於m_ids列表中,如果在說明創建read view時生成該版本的事務還是活躍的,那麼該版本,不可被訪問,如果不在說明創建read view 時生成該版本的事務已經提交,可以被訪問到

如果某個版本數據對當前事務不可見那麼需要一直順着版本鏈找上一個版本的數據,並通過上述步驟判斷是否可見,直到找到可見的版本,如果一直找不到說明該條記錄對當前事務不可見,查詢結果將不包含該記錄。

2.3 Read Committed和 Repeatable Read的不同

  • Read Committed——每次讀取數據前都生成一個Read View

    這樣可以保證生成Read view 中的m_ids是實時活躍事務id集合,也許第一次讀取的時候事務A沒提交,其id位於m_ids中,但是第二次讀取的時候事務A提交了,事務A將不位於m_ids中,這樣在第二次讀取的時候,通過m_ids判斷事務A是否提交的時候,可以得到事務A已經提交了,然後讓事務A版本產生的數據可見(見2.2.4中的內容)。

  • Repeatable Read——如果使用begin開啟事務那麼在第一次查詢的時候生成Read view,如果使用start transaction with consistent snapshot 那麼執行的時候就會生成read view

    這樣可以保證當前事務從頭到尾都是read view中記錄的內容是一致的,第一次讀取的時候事務A沒有提交,那麼不可見,但是第二次讀取的時候事務A提交了,但是read view的m_idsmax_trx_id可以判斷事務A不可見,比如事務A事務id小於max_trx_id意味着生成read view是事務A啟動但是沒提交,即使第二次讀事務A提交了,但是m_ids中還是包含事務A,那麼不可見。如果事務A事務id大於max_trx_id,那麼自然第二次還是大於max_trx_id,也是不可見的,從而實現了可重複讀。

2.4 二級索引與MVCC

上面我們提到,innodb聚簇索引組織的記錄才具備trx_idroll_point,那麼我們使用二級索引進行查詢的時候,如何判斷數據是否可見昵?

  1. 二級索引頁面的page header中存在page_max_trx_id屬性,每當有事務對其中的記錄進行增刪改查操作的時候,如果事務的事務id,大於page_max_trx_id,那麼會更新page_max_trx_id屬性值為其事務id,這意味着page_max_trx_id記錄了修改該二級索引頁面最大的事務id是多少。當select通過二級索引首先看下對於read view的min_trx_id是否大於該頁面的page_max_trx_id,如果大於那麼頁面中所有記錄都對該read view可見,否則就進行下面的第二步
  2. 利用二級索引中的主鍵值,進行回標,得到對應的聚簇索引記錄然後進行回表,然後通過2.2中步驟拿到第一個可見版本的數據,然後比對此紀錄和通過二級索引查詢得到記錄的值是否相同,如果相同那麼發送給客戶端,否則跳過該記錄。
Tags: