首頁 > 軟體

MySQL事務隔離機制詳解

2022-11-23 14:01:43

前言

如何控制並行是資料庫領域中非常重要的問題之一,MySQL為了解決並行帶來的問題,設計了事務隔離機制、鎖機制、MVCC機制,用一整套機制來解決並行問題,本文主要介紹事務隔離機制。

一、什麼是資料庫事務

事務transaction(簡寫tx),在資料庫中,事務是指一組邏輯操作,這些操作要麼全部執行,要麼全部不執行,是一個不可分割的工作單位。事務由事務開始與結束之間執行的全部資料庫操作組成

事務具有以下4大特性,簡稱ACID:

1、原子性(Atomicity)
事務是一個原子操作單元,是不可分割的最小工作單元,就像化學中的原子,其對資料的修改,要麼全都執行,要麼全都不執行

2、一致性(Consistent)
一致性是指事務必須使資料庫從一個一致性狀態變換到另一個一致性狀態。
一致性也稱作是完整性,就是說事務的執行不能破壞資料庫的一致性,在事務開始和完成時,資料都必須保持一致狀態。這意味著所有相關的資料規則都必須應用於事務的修改,以保持資料的完整性

3、隔離性(Isolation)
資料庫提供一定的隔離機制,當多個事務並行存取時,保證事務在不受外部並行操作影響的“獨立”環境執行,當然了,不同的隔離級別決定了有多“獨立”。

4、永續性(Durable)
事務完成之後,它對於資料的修改是永久性的,即使出現系統故障也能夠保持。

小結:原子性,永續性大家基本都能秒懂,隔離性也還好大家稍微想想都能理解,就是接下來要講的隔離級別,但對一致性很多人被官方概念繞住了,其實就是在多事務並行時要保證資料的一致和完整,不理解的話先看下面的問題。

二、事務並行帶來的4類問題

1、髒讀(Dirty Reads)
事務A讀取到了事務B修改但尚未提交的資料,然後事務B回滾了,因此事務A基於此資料做的操作都是無效的,就是讀到了“髒”資料(實際上不應該存在的資料),所以叫髒讀。

2、髒寫或丟失更新(Lost Update)
第一類丟失更新:復原rollback一個事務時,把其他事務已經提交更新的資料回滾掉了。
第二類丟失更新:提交commit一個事務時,把其他事務已經提交更新的資料覆蓋掉了。

3、不可重複讀(Non-Repeatable Reads)
在事務A中,按相同的條件執行查詢過的sql,返回結果發生了修改或被刪除了,這種現象就叫做“不可重複讀”。

4、幻讀(Phantom Reads)
在事務A中,按相同的條件執行查詢過的sql,返回結果中有新資料(是其它事務插入的),就好像發生了幻覺一樣,這就叫“幻讀”。

小結:不可重複讀的和幻讀很容易混淆,不可重複讀側重於修改或刪除,幻讀側重於新增。就這個結論網上很多博主寫的是不可重複讀側重於修改,幻讀側重於新增或刪除 (包括一些知名博主),大家請注意刪除不屬於幻讀,我會在第四段演示時求證。另外後面我會單獨安排一篇介紹MVCC機制,到時你自然會明白這裡的原因。

三、事務4種隔離級別

在SQL標準中定義了四種隔離級別,每一種級別都規定了一個事務中所做的修改,哪些是在事務內和事務間可見的,哪些是不可見的。較低階別的隔離通常可以執行更高的並行,系統的開銷也更低。

每種隔離級別可能出現的問題如下:

事務隔離級別髒讀不可重複讀幻讀
讀未提交(RU)可能可能可能
讀已提交(RC)不可能可能可能
可重複讀(RR)不可能不可能可能
序列化(Serializable)不可能不可能不可能

1、讀未提交(Read uncommitted,RU)

事務中的修改,即使沒有提交,對其他事務也都是可見的,由於可能造成髒讀,所以在實際應用中一般很少使用。

2、讀已提交(Read committed,RC)

事務A修改並提交的資料才會對其它事務可見,所以解決髒讀的問題。大多數資料庫系統的預設隔離級別都是讀已提交(像Oracle、PostgreSQL、SqlServer,但Mysql不是)。

3、可重複讀(Repeatable read,RR)
Mysql預設的事務隔離級別。它保證了在同一個事務中多次讀取相同記錄的結果是一致的,所以解決了髒讀和不可重複讀問題,但未徹底解決幻讀。

4、序列化(Serializable)

可序列化是最高的隔離級別。它通過強制事務序列執行,避免了前面所說的髒讀,不可重複讀和幻讀問題。簡單來說,可序列化會在讀取的每一行資料上都加上鎖,所以可能導致大量的超時和鎖爭用問題。實際應用中也很少用到這個隔離級別,只有在非常需要確保資料的一致性而且可以接受沒有並行的情況下,才考慮用該級別。

四、Mysql演示4種隔離級別

在MySQL中,InnoDB引擎支援事務,MyISAM引擎不支援事務,所以本文實驗的測試環境:Windows 10 + MySQL5.7 + InnoDB。
Mysql5.7可以通過命令select @@tx_isolation;語句來檢視:

mysql> select @@tx_isolation;
+-----------------+
| @@tx_isolation  |
+-----------------+
| REPEATABLE-READ |
+-----------------+

設定4種隔離級別的命令:

set tx_isolation='read-uncommitted';
set tx_isolation='read-committed';
set tx_isolation='repeatable-read';
set tx_isolation='serializable';

開始事務:

begin;

提交事務:

commit;

回滾事務:

rollback;

1、讀未提交(RU)

set tx_isolation='read-uncommitted';
select @@tx_isolation;
begin;

驗證髒讀

2、讀已提交(RC)

set tx_isolation='read-committed';
select @@tx_isolation;
begin;

驗證解決髒讀,存在不可重複讀

驗證幻讀

3、可重複讀(RR)

set tx_isolation='repeatable-read';
select @@tx_isolation;
begin;

驗證解決髒讀和不可重複讀
上面說的不可重複讀側重於修改或刪除,求證請看下面的截圖,

驗證幻讀
Mysql的RR級別在一定程度上解決了幻讀問題,但不夠徹底,至於為什麼這麼說?還是那句話,得深入理解MVCC機制,這個會單獨安排一篇,敬請期待!

4、序列化(Serializable)

set tx_isolation='serializable';
select @@tx_isolation;
begin;

如果長時間阻塞,最後會鎖等待超時,丟擲異常:ERROR 1205 (HY000): Lock wait timeout exceeded; try restarting transaction
換成是在事務A執行,結果也是一樣會阻塞,直到其它事務完成或超時丟擲同樣的異常。
對於update也是同樣的道理,詳細如下圖:

總結

通過本文我們已經詳細說明了:

事務4大特性:ACID事務並行帶來的4類問題:髒讀、髒寫、不可重複讀、幻讀事務的4種隔離級別:讀未提交、讀已提交、可重複讀、序列化Mysql提供的4種隔離級別演示

剩下的MySQL鎖機制和MVCC機制,我們下文見!

參考文章:

什麼是事務?
三分鐘圖解事務隔離級別,看一遍就懂
MySQL事務隔離級別詳解
MySQL的四種事務隔離級別
透徹解讀mysql的可重複讀、幻讀及實現原理
【MySQL (三) | 五分鐘搞清楚MySQL事務隔離級別】

到此這篇關於MySQL事務隔離機制的文章就介紹到這了,更多相關mysql事務隔離內容請搜尋it145.com以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援it145.com!


IT145.com E-mail:sddin#qq.com