一起聊聊MySQL事務與MVCC如何實現的隔離級別
這篇文章為大家帶來了關於MySQL事務與MVCC如何實現的隔離等級的相關問題,希望對大家有幫助。
資料庫事務介紹
交易的四大特性(ACID)
原子性(atomicity): 事務的最小工作單元,要麼全成功,要麼全失敗。
一致性(consistency): 事務開始和結束後,資料庫的完整性不會被破壞。
隔離性(isolation): 不同交易之間互不影響,四種隔離等級為RU(讀取未提交)、RC(讀取已提交) 、RR(可重複讀)、SERIALIZABLE (串列化)。
持久性(durability): 事務提交後,資料的修改是永久性的,即使系統故障也不會遺失。
交易的隔離等級
已讀取未提交(Read UnCommitted/RU)
又稱為髒讀,一個事務可以讀取到另一個事務未提交的資料。這種隔離等級歲最不安全的一種,因為未提交的交易是存在回滾的情況。
讀取已提交(Read Committed/RC)
又稱為不可重複讀取,一個交易因為讀取到另一個交易已提交的修改數據,導致在當前事務的不同時間讀取同一數據獲取的結果不一致。
舉個例子,在下面的例子中就會發現SessionA在一個交易期間兩次查詢的資料不一樣。原因是在於目前隔離等級為 RC,SessionA的事務可以讀取到SessionB提交的最新資料。
發生時間 | #SessionA | SessionB |
---|---|---|
1 | begin; | |
#2 | select * from user where id=1;(張三) | |
3 | update user set name='李四' where id=1;(預設隱式提交交易) | |
4 | select * from user where id =1;(李四) | |
5 | update user set name='王二' where id=1;(預設隱式提交交易) | |
6 | select * from user where id=1;(王二) |
可重複讀取(Repeatable Read/RR)
又稱為幻讀,一個事物讀取可以讀取到其他事務提交的數據,但是在RR隔離級別下,當前讀取此條數據只可讀取一次,在當前事務中,不論讀取多少次,數據任然是第一次讀取的值,不會因為在第一次讀取之後,其他交易再修改提交此資料而產生改變。因此也成為幻讀,因為讀出來的數據不一定就是最新的數據。
舉個例子:在SessionA中第一次讀取數據時,後續其他事務修改提交數據,不會再影響到SessionA讀取的數據值。此為可重複讀取。
發生時間 | #SessionA | SessionB |
---|---|---|
1 | begin; | |
#2 | select * from user where id=1;(張三) | |
3 | update user set name='李四' where id=1; (預設隱式提交交易) | |
4 | select * from user where id =1;(張三) | |
5 | update user set name='王二' where id=1;(預設隱式提交交易) | |
6 | select * from user where id=1;(張三) |
#序列化(Serializable)
所有的資料庫的讀取或寫入操作都是串行執行,當前隔離等級下只支援單一請求同時執行,所有的操作都需要佇列執行。所以種隔離等級下所有的資料是最穩定的,但是效能也是最差的。資料庫的鎖定實作就是這種隔離等級的更小粒度版本。
發生時間 | #SessionA | SessionB | |
---|---|---|---|
1 | begin; | ||
#2 | begin; | ||
3 | update user set name='李四' where id=1; | ||
4 | select * from user where id=1;(等待、wait) | ||
#5 | commit; | ||
select * from user where id=1;(李四) |
發生時間 | SessionA | SessionB |
---|---|---|
1 | begin; | |
2 | begin; | |
3 | 查詢餘額= 1000元 | |
4 | 查詢餘額= 1000元 | |
5 | #存入金額100元,修改餘額為1100元 | |
6 | 取出現金100元,此時修改餘額為900元 | |
提交交易(餘額= 900) |
#發生時間 | ||
---|---|---|
SessionB | ||
#begin; |
||
2 |
||
3 | ||
#4 |
查詢餘額= 1000元 | |
#5 | ||
6 |
取出現金100元,此時修改餘額為900元 | |
8 |
上面的兩種情況就是對於一條數據,多個事務同時操作可能會產生的問題,會出現某個事務的操作被覆蓋而導致數據丟失。
LBCC 解決資料遺失
LBCC,基於鎖定的並發控制,Lock Based Concurrency Control。
使用鎖的機制,在當前事務需要對資料修改時,將當前事務加上鎖,同一個時間只允許一條事務修改當前數據,其他事務必須等待鎖釋放之後才可以操作。
MVCC 解決資料遺失
MVCC,多重版本的並發控制,Multi-Version Concurrency Control。
使用版本來控制並發情況下的資料問題,在B事務開始修改帳戶且事務未提交時,當A事務需要讀取帳戶餘額時,此時會讀取到B事務修改操作之前的帳戶餘額的副本數據,但是如果A事務需要修改帳戶餘額數據就必須要等待B事務提交事務。
MVCC使得資料庫讀取不會對資料加鎖,普通的SELECT請求不會加鎖,提高了資料庫的並發處理能力。借助MVCC,資料庫可以實現READ COMMITTED,REPEATABLE READ等隔離級別,使用者可以查看當前資料的前一個或前幾個歷史版本,保證了ACID中的I特性(隔離性)。
InnoDB的MVCC實作邏輯
InnoDB儲存引擎所保存的MVCC的資料
InnoDB的MVCC是透過在每行記錄後面保存兩個隱藏的列來實現的。 一個儲存了行的交易ID(DB_TRX_ID),一個儲存了行的回滾指標(DB_ROLL_PT)。每開始一個新的事務,就會自動遞增產 生一個新的事務id。事務開始時刻的會把事務id放到目前事務影響的行事務id中,當查詢時需要用當前事務id和每行記錄的事務id進行比較。
下面看一下在REPEATABLE READ隔離等級下,MVCC具體是如何操作的。
SELECT
InnoDB 會根據以下兩個條件檢查每行記錄:
InnoDB只查找版本早於當前事務版本的資料行(也就是,行的事務編號小於或等於目前事務的事務編號),這樣可以確保事務讀取的行,要麼是在事務開始前已經存在的,要麼是事務本身插入或修改過的。
刪除的行要事務ID判斷,讀取到事務開始之前狀態的版本,只有符合上述兩個條件的記錄,才能傳回作為查詢結果。
INSERT
InnoDB為新插入的每一行儲存目前交易編號作為行版本號。
DELETE
InnoDB為刪除的每一行儲存目前交易編號作為行刪除標識。
UPDATE
InnoDB為插入一行新記錄,保存目前交易編號作為行版本號,同時保存目前交易編號到原來的行作為行刪除標識。
儲存這兩個額外交易編號,讓大多數讀取作業都可以不用加鎖。這樣設計使得讀取資料操作很簡單,效能很好,也能保證只會讀取到符合標準的行。不足之處是每行記錄都需要額外的儲存空間,需要做更多的行檢查工作,以及一些額外的維護工作。
MVCC只在REPEATABLE READ和READ COMMITIED兩個隔離等級下運作。其他兩個隔離等級都和 MVCC不相容 ,因為READ UNCOMMITIED總是讀取最新的資料行,而不是符合目前事務版本的資料行。而SERIALIZABLE則會對所有讀取的行都加鎖。
MVCC 在mysql 中的實作依賴的是 undo log 與 read view 。
undo log
依行為的不同,undo log分為兩種:insert undo log 和 update undo log
#insert undo log:
#insert 操作中產生的undo log,因為insert操作記錄只對目前交易本身課件,對於其他事務此記錄不可見,所以insert undo log 可以在事務提交後直接刪除而不需要進行purge操作。
purge的主要任務是將資料庫中已經mark del 的資料刪除,另外也會大量回收undo pages
資料庫Insert時的資料初始狀態:
-
update undo log:
update 或delete 操作中產生的undo log。因為會對已經存在的記錄產生影響,為了提供 MVCC機制,因此update undo log 不能在交易提交時就進行刪除,而是將事務提交時放到入 history list 上,等待 purge 執行緒進行最後的刪除操作。
資料第一次修改時:
#當另一個交易第二次修改目前資料:
為了確保交易並發操作時,在寫各自的undo log時不產生衝突,InnoDB採用回滾段的方式來維護undo log的並發寫入和持久化。回滾段其實是一種 Undo 檔案組織方式。
ReadView
#對於 RU(READ UNCOMMITTED) 隔離級別下,所有事務直接讀取資料庫的最新值即可,和 SERIALIZABLE 隔離級別,所有請求都會加鎖,同步執行。所以這對這兩種情況是不需要使用到 Read View# 的版本控制。
對於 RC(READ COMMITTED) 和 RR(REPEATABLE READ) 隔離等級的實作就是透過上面的版本控制來完成。兩種隔離界別下的核心處理邏輯就是判斷所有版本中哪個版本是目前事務可見的處理。針對這個問題InnoDB在設計上增加了ReadView的設計,ReadView中主要包含當前系統中還有哪些活躍的讀寫事務,把它們的事務id放到一個列表中,我們把這個清單命名為為m_ids。
對於查詢時的版本鏈資料是否看見的判斷邏輯:
如果被存取版本的trx_id 屬性值小於m_ids 清單中最小的事務id,表示生成該版本的事務在產生ReadView 前已經提交,所以該版本可以被目前事務存取。
如果被存取版本的trx_id 屬性值大於m_ids 清單中最大的事務id,表示生成該版本的事務在生成ReadView 後才生成,所以該版本不可以被目前事務訪問。
如果被存取版本的trx_id 屬性值在m_ids 清單中最大的交易id和最小交易id之間,那就需要判斷一下trx_id 屬性值是不是在m_ids 清單中,如果在,說明建立ReadView 時產生該版本的事務還是活躍的,則該版本不可以被存取;如果不在,說明在建立ReadView 時產生該版本的交易已被提交,該版本可以被存取。
舉例:
READ COMMITTED 隔離等級下的ReadView
每次讀取資料前都產生一個ReadView (m_ids列表)
#Transaction 777 | #Transaction 888 | Trasaction 999 | |
---|---|---|---|
# begin; | |||
##begin; |
begin; | ||
UPDATE user SET name = 'CR7' WHERE id = 1; | |||
#... |
|||
UPDATE user SET name = 'Messi' WHERE id = 1; |
SELECT * FROM user where id = 1; |
||
commit; |
| ##T7||
UPDATE user SET name = 'Neymar' WHERE id = 1; |
|||
SELECT * FROM user where id = 1; |
T9 | ||
UPDATE user SET name = 'Dybala' WHERE id = 1; |
|||
commit; |
|||
這裡分析下上面的情況下的ReadView
時間點T5 情況下的SELECT 語句:
目前時間點的版本鏈:
##此時SELECT 語句執行,目前資料的版本鏈如上,因為目前的事務777,和事務888 都未提交,所以此時的活躍事務的ReadView的列表情況時間點T8 情況下的SELECT 語句:目前時間的版本鏈情況:m_ids:[777, 888] ,因此查詢語句會根據目前版本鏈中小於 m_ids# 中的最大的版本數據,即查詢到的是Mbappe。
此時SELECT 語句執行,目前資料的版本鏈如上,因為目前的事務777已經提交,和事務888 未提交,所以此時的活躍事務的ReadView的列表情況時間點T11 情況下的SELECT 語句:目前時間點的版本鏈資訊:m_ids:[888] ,因此查詢語句會根據當前版本鏈中小於m_ids 中的最大的版本數據,即查詢到的是Messi。
此時SELECT 語句執行,目前資料的版本鍊如上,因為目前的事務777和事務888 都已經提交,所以此時的活躍事務的ReadView的列表為空,因此查詢語句會直接查詢當前資料庫最新數據,即查詢到的是Dybala。
總結: 使用READ COMMITTED隔離等級的交易在每次查詢開始時都會產生一個獨立的 ReadView。
REPEATABLE READ 隔離等級下的ReadView#在交易開始後第一次讀取資料時產生一個ReadView(m_ids清單)
時間 | Transaction 777 | Transaction 888 | #Trasaction 999 |
---|---|---|---|
begin; | |||
##T2 |
|||
#begin; | begin; |
||
T3 |
UPDATE user SET name = 'CR7' WHERE id = 1; |
||
T4 |
|||
#UPDATE user SET name = 'Messi' WHERE id = 1; | |||
T6 |
commit; |
||
T7 |
|||
|
|||
##T9
UPDATE user SET name = 'Dybala' WHERE id = 1;
T10
commit;
T11
以上是一起聊聊MySQL事務與MVCC如何實現的隔離級別的詳細內容。更多資訊請關注PHP中文網其他相關文章!

熱AI工具

Undresser.AI Undress
人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover
用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool
免費脫衣圖片

Clothoff.io
AI脫衣器

Video Face Swap
使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱門文章

熱工具

記事本++7.3.1
好用且免費的程式碼編輯器

SublimeText3漢化版
中文版,非常好用

禪工作室 13.0.1
強大的PHP整合開發環境

Dreamweaver CS6
視覺化網頁開發工具

SublimeText3 Mac版
神級程式碼編輯軟體(SublimeText3)

HQL和SQL在Hibernate框架中進行比較:HQL(1.物件導向語法,2.資料庫無關的查詢,3.類型安全),而SQL直接操作資料庫(1.與資料庫無關的標準,2.可執行複雜查詢和資料操作)。

《OracleSQL中除法運算的用法》在OracleSQL中,除法運算是常見的數學運算之一。在資料查詢和處理過程中,除法運算可以幫助我們計算欄位之間的比例或得出特定數值的邏輯關係。本文將介紹OracleSQL中除法運算的用法,並提供具體的程式碼範例。一、OracleSQL中除法運算的兩種方式在OracleSQL中,除法運算可以用兩種不同的方式來進行

Oracle和DB2是兩個常用的關聯式資料庫管理系統,它們都有自己獨特的SQL語法和特性。本文將針對Oracle和DB2的SQL語法進行比較與區別,並提供具體的程式碼範例。資料庫連接在Oracle中,使用以下語句連接資料庫:CONNECTusername/password@database而在DB2中,連接資料庫的語句如下:CONNECTTOdataba

MyBatis動態SQL標籤解讀:Set標籤用法詳解MyBatis是一個優秀的持久層框架,它提供了豐富的動態SQL標籤,可以靈活地建構資料庫操作語句。其中,Set標籤是用來產生UPDATE語句中SET子句的標籤,在更新作業中非常常用。本文將詳細解讀MyBatis中Set標籤的用法,以及透過具體的程式碼範例來示範其功能。什麼是Set標籤Set標籤用於MyBati

SQL中的Identity是什麼,需要具體程式碼範例在SQL中,Identity是一種用於產生自增數字的特殊資料類型,它常用於唯一識別表中的每一行資料。 Identity欄位通常與主鍵列搭配使用,可確保每筆記錄都有獨一無二的識別碼。本文將詳細介紹Identity的使用方式以及一些實際的程式碼範例。 Identity的基本使用方式在建立表格時,可以使用Identit

在Springboot+Mybatis-plus不使用SQL語句進行多表添加操作我所遇到的問題準備工作在測試環境下模擬思維分解一下:創建出一個帶有參數的BrandDTO對像模擬對後台傳遞參數我所遇到的問題我們都知道,在我們使用Mybatis-plus中進行多表操作是極其困難的,如果你不使用Mybatis-plus-join這一類的工具,你只能去配置對應的Mapper.xml文件,配置又臭又長的ResultMap,然後再寫對應的sql語句,這種方法雖然看上去很麻煩,但具有很高的靈活性,可以讓我們

解決方法:1、檢查登入使用者是否具有足夠的權限來存取或操作該資料庫,確保該使用者俱有正確的權限;2、檢查SQL Server服務的帳戶是否具有存取指定檔案或資料夾的權限,確保該帳戶具有足夠的權限來讀取和寫入該文件或資料夾;3、檢查指定的資料庫文件是否已被其他進程打開或鎖定,嘗試關閉或釋放該文件,並重新運行查詢;4、嘗試以管理員身份運行Management Studio等等。

如何使用SQL語句在MySQL中進行資料聚合和統計?在進行資料分析和統計時,資料聚合和統計是非常重要的步驟。 MySQL作為一個功能強大的關聯式資料庫管理系統,提供了豐富的聚合和統計函數,可以很方便地進行資料聚合和統計操作。本文將介紹使用SQL語句在MySQL中進行資料聚合和統計的方法,並提供具體的程式碼範例。一、使用COUNT函數進行計數COUNT函數是最常用
