MySQL事務隔離

事務隔離級別

SQL標準定義了4類隔離級別,包括了一些具體規則,用來限定事務內外的哪些改變是可見的,哪些是不可見的。低級別的隔離級一般支持更高的併發處理,並擁有更低的系統開銷。

  • Read Uncommitted(讀取未提交內容)

在該隔離級別,所有事務都可以"看到"其他未提交事務的執行結果。本隔離級別很少用於實際應用,因爲它的性能也不比其他級別好多少。讀取未提交的數據,也被稱之爲髒讀(Dirty Read)。

  • Read Committed(讀取提交內容)

大多數數據庫系統的默認隔離級別(但不是MySQL默認的)。它滿足了隔離的簡單定義:一個事務在開始時,只能“看見”已經提交事務所做的改變,一個事務從開始到提交前,所做的任何數據改變都是不可見的,除非已經提交。這種隔離級別也支持所謂的不可重複讀(Nonrepeatable Read),因爲同一事務的其他實例在該實例處理期間可能會有新的commit,所以同一select可能返回不同結果。

  • Repeatable Read(可重讀)

MySQL的默認事務隔離級別,它確保同一事務的多個實例在併發讀取數據時,會“看到同樣的“數據行。不過理論上,這會導致另一個棘手的問題:幻讀 (Phantom Read)。簡單的說,幻讀指當用戶讀取某一範圍的數據行時,另一個事務又在該範圍內插入了新行,當用戶再讀取該範圍的數據行時,會發現有新的“幻影” 行。InnoDB和Falcon存儲引擎通過多版本併發控制(MVCC,Multiversion Concurrency Control)機制解決了該問題。

  • Serializable(可串行化)

這是最高的隔離級別,它通過強制事務排序,使之不可能相互衝突,從而解決幻讀問題。簡言之,它是在每個讀的數據行上加上共享鎖。在這個級別,可能導致大量的超時現象和鎖競爭。

這四種隔離級別採取不同的鎖類型來實現,若讀取的是同一個數據的話,就容易發生問題。例如:

  • 髒讀(Drity Read):某個事務已更新一份數據,另一個事務在此時讀取了同一份數據,由於某些原因,前一個RollBack了操作,則後一個事務所讀取的數據就會是不正確的。

  • 不可重複讀(Non-repeatable read):在一個事務的兩次查詢之中數據不一致,這可能是兩次查詢過程中間插入了一個事務更新的原有的數據。

  • 幻讀(Phantom Read):在一個事務的兩次查詢中數據筆數不一致,例如有一個事務查詢了幾列(Row)數據,而另一個事務卻在此時插入了新的幾列數據,先前的事務在接下來的查詢中,就會發現有幾列數據是它先前所沒有的。

隔離級別 髒讀可能性 不可重複讀可能性 幻讀可能性 加鎖讀
Read Uncommitted
Read Committed
Repeatable Read
Serializable

修改事務隔離級別

查看當前數據庫事務隔離級別
mysql> SELECT @@GLOBAL.tx_isolation, @@tx_isolation;
+-----------------------+-----------------+
| @@GLOBAL.tx_isolation | @@tx_isolation  |
+-----------------------+-----------------+
| REPEATABLE-READ       | REPEATABLE-READ |
+-----------------------+-----------------+
1 row in set (0.00 sec)

修改當前數據庫事務隔離級別
#可選參數 READ-UNCOMMITTED, READ-COMMITTED, REPEATABLE-READ, SERIALIZABLE
[mysqld]
transaction-isolation = REPEATABLE-READ
在線修改
mysql> SET GLOBAL tx_isolation='READ-UNCOMMITTED';
Query OK, 0 rows affected (0.00 sec)

mysql> SELECT @@GLOBAL.tx_isolation, @@tx_isolation;
+-----------------------+-----------------+
| @@GLOBAL.tx_isolation | @@tx_isolation  |
+-----------------------+-----------------+
| READ-UNCOMMITTED      | REPEATABLE-READ |
+-----------------------+-----------------+
1 row in set (0.00 sec)

READ-UNCOMMITTED(讀取未提交內容)

1. 查看隔離級別:
mysql> select @@global.tx_isolation,@@tx_isolation;  
+-----------------------+------------------+
| @@global.tx_isolation | @@tx_isolation   |
+-----------------------+------------------+
| READ-UNCOMMITTED      | READ-UNCOMMITTED |
+-----------------------+------------------+
1 row in set (0.00 sec)

2. 關閉事務自動提交
mysql> set session autocommit=off;
Query OK, 0 rows affected (0.01 sec)

3. 在A session開始事務並查看數據:
mysql> start transaction;  
Query OK, 0 rows affected (0.00 sec)

mysql> select * from test;
+----+------+
| id | name |
+----+------+
|  1 | 1    |
|  2 | 2    |
|  3 | 3    |
+----+------+
3 rows in set (0.00 sec)

4. 在B session開始事務並插入數據:
mysql> start transaction; 
Query OK, 0 rows affected (0.01 sec)

mysql> insert into test values(4,4),(5,5);
Query OK, 2 rows affected (0.01 sec)
Records: 2  Duplicates: 0  Warnings: 0

5. 在A session查看數據:
mysql> select * from test;
+----+------+
| id | name |
+----+------+
|  1 | 1    |
|  2 | 2    |
|  3 | 3    |
|  4 | 4    |
|  5 | 5    |
+----+------+
5 rows in set (0.00 sec)

6. 在B session回滾事務:
mysql> rollback;
Query OK, 0 rows affected (0.05 sec)

7. 在A session查看數據:
mysql> select * from test.test;
+----+------+
| id | name |
+----+------+
|  1 | 1    |
|  2 | 2    |
|  3 | 3    |
+----+------+
3 rows in set (0.01 sec)

經過上面的測試可以看出,事務B更新了一條記錄,但是沒有提交,此時事務A可以查詢出未提交記錄,但一旦事務B回滾後,事務A兩次查詢的數據不一樣,造成髒讀現象。未提交讀是最低的隔離級別。

Read Committed(讀取提交內容)

修改事務級別
mysql> SELECT @@GLOBAL.tx_isolation, @@tx_isolation;
+-----------------------+----------------+
| @@GLOBAL.tx_isolation | @@tx_isolation |
+-----------------------+----------------+
| READ-COMMITTED        | READ-COMMITTED |
+-----------------------+----------------+
1 row in set (0.00 sec)

1. 在A session開始事務並查看數據:
mysql> start transaction;  
Query OK, 0 rows affected (0.00 sec)

mysql> select * from test;  
+----+------+
| id | name |
+----+------+
|  1 | 1    |
|  2 | 2    |
|  3 | 3    |
+----+------+
3 rows in set (0.00 sec)

2. 在B session開始事務、更新數據:
mysql> start transaction;  
Query OK, 0 rows affected (0.00 sec)

mysql> select * from test;
+----+------+
| id | name |
+----+------+
|  1 | 1    |
|  2 | 2    |
|  3 | 3    |
+----+------+
3 rows in set (0.00 sec)

mysql> delete from test where id = 3;
Query OK, 1 row affected (0.04 sec)

mysql> select * from test;
+----+------+
| id | name |
+----+------+
|  1 | 1    |
|  2 | 2    |
+----+------+
2 rows in set (0.00 sec)

3. 在A session查看數據:
mysql> select * from test;
+----+------+
| id | name |
+----+------+
|  1 | 1    |
|  2 | 2    |
|  3 | 3    |
+----+------+
3 rows in set (0.00 sec)

4. 在B session提交事務:
mysql> commit;
Query OK, 0 rows affected (0.06 sec)

5. 在A session查看數據:
mysql> select * from test; 
+----+------+
| id | name |
+----+------+
|  1 | 1    |
|  2 | 2    |
+----+------+
2 rows in set (0.01 sec)

經過上面的實例可以得出結論,已提交讀隔離級別解決了髒讀的問題,但是出現了不可重複讀的問題,即事務A在兩次查詢的數據不一致,因爲在兩次查詢之間事務B更新了一條數據。已提交讀級別只允許讀取已提交的記錄,但不要求可重複讀。

Repeatable Read(可重讀)

修改事務級別
mysql> SELECT @@GLOBAL.tx_isolation, @@tx_isolation;
+-----------------------+-----------------+
| @@GLOBAL.tx_isolation | @@tx_isolation  |
+-----------------------+-----------------+
| REPEATABLE-READ       | REPEATABLE-READ |
+-----------------------+-----------------+
1 row in set (0.01 sec)

1. 在A session開始事務並查看數據:
mysql> start transaction;  
Query OK, 0 rows affected (0.00 sec)

mysql> select * from test;  
+----+------+
| id | name |
+----+------+
|  1 | 1    |
|  2 | 2    |
+----+------+
2 rows in set (0.00 sec)

2. 在B session開始事務、更新數據:
mysql> start transaction;  
Query OK, 0 rows affected (0.00 sec)

mysql> select * from test;
+----+------+
| id | name |
+----+------+
|  1 | 1    |
|  2 | 2    |
+----+------+
2 rows in set (0.00 sec)

mysql> insert into test values(3,3);
Query OK, 1 row affected (0.00 sec)

mysql> select * from test;
+----+------+
| id | name |
+----+------+
|  1 | 1    |
|  2 | 2    |
|  3 | 3    |
+----+------+
3 rows in set (0.00 sec)

3. 在A session查看數據:
mysql> select * from test; 
+----+------+
| id | name |
+----+------+
|  1 | 1    |
|  2 | 2    |
+----+------+
2 rows in set (0.00 sec)

4. 在B session提交事務:
mysql> commit;
Query OK, 0 rows affected (0.06 sec)

mysql> select * from test;
+----+------+
| id | name |
+----+------+
|  1 | 1    |
|  2 | 2    |
|  3 | 3    |
+----+------+
3 rows in set (0.00 sec)

5. 在A session查看數據:
mysql> select * from test; 
+----+------+
| id | name |
+----+------+
|  1 | 1    |
|  2 | 2    |
+----+------+
2 rows in set (0.01 sec)

6. 在A session提交事務並查看數據:
mysql> commit;
Query OK, 0 rows affected (0.00 sec)

mysql> select * from test;
+----+------+
| id | name |
+----+------+
|  1 | 1    |
|  2 | 2    |
|  3 | 3    |
+----+------+
3 rows in set (0.00 sec)

由以上的實例可以得出結論,可重複讀隔離級別只允許讀取已提交記錄,而且在一個事務兩次讀取一個記錄期間數據保持一致,但該事務不要求與其他事務可串行化。例如,當一個事務可以找到由一個已提交事務更新的記錄,但是可能產生幻讀問題(注意是可能,因爲數據庫對隔離級別的實現有所差別)。像以上的實例,就沒有出現數據幻讀的問題。

Serializable(可串行化)

修改事務級別
mysql> SELECT @@GLOBAL.tx_isolation, @@tx_isolation;
+-----------------------+----------------+
| @@GLOBAL.tx_isolation | @@tx_isolation |
+-----------------------+----------------+
| SERIALIZABLE          | SERIALIZABLE   |
+-----------------------+----------------+
1 row in set (0.00 sec)

1. 在A session開始事務並查看數據:
mysql> start transaction; 
Query OK, 0 rows affected (0.01 sec)

mysql> select * from test; 
+----+------+
| id | name |
+----+------+
|  1 | 1    |
|  2 | 2    |
|  3 | 3    |
+----+------+
3 rows in set (0.01 sec)

2. 在B session開始事務並查看更新數據:
mysql> start transaction; 
Query OK, 0 rows affected (0.01 sec)

mysql> select * from test;
+----+------+
| id | name |
+----+------+
|  1 | 1    |
|  2 | 2    |
|  3 | 3    |
+----+------+
3 rows in set (0.00 sec)

# 事務A一直不提交,事務B超時  
mysql> insert into test values(4,4);
ERROR 1205 (HY000): Lock wait timeout exceeded; try restarting transaction

3. 在A session提交事務:
mysql> commit;
Query OK, 0 rows affected (0.00 sec)

4. 在B session更新數據:
mysql> insert into test values(4,4);
Query OK, 1 row affected (0.00 sec)

5. 在A session查看數據:
# 事務B一直不提交,事務A超時 
mysql> select * from test;
ERROR 1205 (HY000): Lock wait timeout exceeded; try restarting transaction

從實例中可以看出serializable完全鎖定字段,若一個事務來查詢同一份數據就必須等待,直到前一個事務完成並解除鎖定爲止 。是完整的隔離級別,會鎖定對應的數據表格,因而會有效率的問題。

整理自網絡

Svoid
2015-03-05

發佈了54 篇原創文章 · 獲贊 5 · 訪問量 3萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章