SELECT @@session.tx_isolation;
SELECT @@tx_isolation;
複製代碼
SELECT @@global.tx_isolation;
複製代碼
transaction-isolation = {READ-UNCOMMITTED | READ-COMMITTED | REPEATABLE-READ | SERIALIZABLE}
複製代碼
SET [SESSION | GLOBAL] TRANSACTION ISOLATION LEVEL {READ UNCOMMITTED | READ COMMITTED | REPEATABLE READ | SERIALIZABLE}
複製代碼
隔離級別 | 髒讀(Dirty Read) | 不可重複讀(NonRepeatable Read) | 幻讀(Phantom Read) |
---|---|---|---|
未提交讀(Read uncommitted) | 可能 | 可能 | 可能 |
已提交讀(Read committed) | 不可能 | 可能 | 可能 |
可重複讀(Repeatable read) | 不可能 | 不可能 | 可能 |
可串行化(Serializable ) | 不可能 | 不可能 | 不可能 |
CREATE TABLE `test` (
`id` bigint unsigned NOT NULL AUTO_INCREMENT COMMENT 'id',
`name` varchar(50) NOT NULL DEFAULT '' COMMENT '名字',
PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COMMENT='測試';
複製代碼
mysql> SELECT @@session.tx_isolation; // 查詢會話隔離級別可重複讀
+------------------------+
| @@session.tx_isolation |
+------------------------+
| REPEATABLE-READ |
+------------------------+
1 row in set, 1 warning (0.04 sec)
mysql> SELECT @@tx_isolation; //查詢系統隔離級別爲可重複讀
+-----------------+
| @@tx_isolation |
+-----------------+
| REPEATABLE-READ |
+-----------------+
1 row in set, 1 warning (0.00 sec)
mysql> start transaction; //開啓事務
Query OK, 0 rows affected (0.00 sec)
mysql> insert into test(`name`) values("qiu"); //插入數據成功,此時事務尚未提交
Query OK, 1 row affected (0.01 sec)
mysql> select * from test;
+----+------+
| id | name |
+----+------+
| 1 | qiu |
+----+------+
1 row in set (0.00 sec)
複製代碼
mysql> SELECT @@session.tx_isolation; //會話隔離級別爲可重複讀
+------------------------+
| @@session.tx_isolation |
+------------------------+
| REPEATABLE-READ |
+------------------------+
1 row in set, 1 warning (0.00 sec)
mysql> SELECT @@tx_isolation; //系統隔離級別爲可重複讀
+-----------------+
| @@tx_isolation |
+-----------------+
| REPEATABLE-READ |
+-----------------+
1 row in set, 1 warning (0.00 sec)
mysql> select * from test; //查詢不到 sessoin1 未提交的數據,不會出現髒讀現象
Empty set (0.00 sec)
複製代碼
mysql> SELECT @@session.tx_isolation;//會話隔離級別爲未提交讀
+------------------------+
| @@session.tx_isolation |
+------------------------+
| READ-UNCOMMITTED | --------讀到了 session1 未提交的數據,出現髒讀現象
+------------------------+
1 row in set, 1 warning (0.00 sec)
mysql> select * from test;//讀到了 session1 未提交的數據,此爲髒讀
+----+------+
| id | name |
+----+------+
| 1 | qiu |
+----+------+
1 row in set (0.00 sec)
複製代碼
在同一個事務內,屢次讀取同一個數據,此時事務尚未完成。另外一個事務在前一個事務兩次讀取之間修改了數據,因爲修改了數據,前一個事務讀到的數據不同,所以稱爲不可重複讀。html
sessoin1(事務內第一次讀)mysql
mysql> SELECT @@session.tx_isolation; //隔離級別爲提交讀
+------------------------+
| @@session.tx_isolation |
+------------------------+
| READ-COMMITTED |
+------------------------+
1 row in set, 1 warning (0.00 sec)
mysql> start transaction; //開啓事務
Query OK, 0 rows affected (0.00 sec)
mysql> select * from test; //開啓事務內的第一次查詢
+----+------+
| id | name |
+----+------+
| 2 | qiu |
+----+------+
1 row in set (0.01 sec)
複製代碼
mysql> SELECT @@session.tx_isolation;//隔離級別爲可重複讀
+------------------------+
| @@session.tx_isolation |
+------------------------+
| REPEATABLE-READ |
+------------------------+
1 row in set, 1 warning (0.00 sec)
mysql> start transaction; //開啓事務
Query OK, 0 rows affected (0.00 sec)
mysql> select * from test;
+----+------+
| id | name |
+----+------+
| 2 | qiu |
+----+------+
1 row in set (0.01 sec)
mysql> insert into test(`name`) values ("hello"); //在sessoin1第一次查詢後修改了數據
Query OK, 1 row affected (0.01 sec)
mysql> select * from test;
+----+-------+
| id | name |
+----+-------+
| 2 | qiu |
| 3 | hello |
+----+-------+
2 rows in set (0.00 sec)
mysql> commit; //提交事務
Query OK, 0 rows affected (0.01 sec)
複製代碼
mysql> select * from test; //在事務內第二次讀,讀到了 sessoin2 提交的數據
+----+-------+
| id | name |
+----+-------+
| 2 | qiu | ---------------READ-COMMITTED級別出現不可重複讀現象
| 3 | hello |
+----+-------+
2 rows in set (0.00 sec)
複製代碼
mysql> SELECT @@session.tx_isolation;//隔離級別爲可重複讀
+------------------------+
| @@session.tx_isolation |
+------------------------+
| REPEATABLE-READ |
+------------------------+
1 row in set, 1 warning (0.00 sec)
mysql> start transaction; //開啓事務
Query OK, 0 rows affected (0.00 sec)
mysql> select * from test;
+----+-------+
| id | name |
+----+-------+
| 2 | qiu |
| 3 | hello |
+----+-------+
2 rows in set (0.00 sec)
複製代碼
mysql> SELECT @@session.tx_isolation; //隔離級別爲可重複讀
+------------------------+
| @@session.tx_isolation |
+------------------------+
| REPEATABLE-READ |
+------------------------+
1 row in set, 1 warning (0.00 sec)
mysql> start transaction;
Query OK, 0 rows affected (0.00 sec)
mysql> select * from test;
+----+-------+
| id | name |
+----+-------+
| 2 | qiu |
| 3 | hello |
+----+-------+
2 rows in set (0.00 sec)
mysql> insert into test (`name`) values ("hi"); //sessoin1 第一次讀以後改變數據
Query OK, 1 row affected (0.01 sec)
mysql> select * from test;
+----+-------+
| id | name |
+----+-------+
| 2 | qiu |
| 3 | hello |
| 4 | hi |
+----+-------+
3 rows in set (0.00 sec)
mysql> commit; //提交事務
Query OK, 0 rows affected (0.00 sec)
複製代碼
mysql> select * from test;
+----+-------+
| id | name |
+----+-------+
| 2 | qiu |
| 3 | hello | -------------sessoin1 沒有讀到 sessoin2 提交的數據,出現可重複讀現象
+----+-------+
2 rows in set (0.00 sec)
複製代碼
第一個事務對錶中的全部數據進行修改,第二個事務往表裏面插入一條數據。此時第一個事務發現表中還有未修改的數據,好像出現了幻覺同樣。sql
幻讀現象1:數據庫
session1: session2:
mysql> select @@global.tx_isolation, @@tx_isolation;
+-----------------------+-----------------+
| @@global.tx_isolation | @@tx_isolation |
+-----------------------+-----------------+
| REPEATABLE-READ | REPEATABLE-READ |
+-----------------------+-----------------+
1 row in set, 2 warnings (0.00 sec)
mysql> start transaction;
Query OK, 0 rows affected (0.00 sec)
mysql> start transaction;
Query OK, 0 rows affected (0.00 sec)
mysql> select * from test;
Empty set (0.00 sec)
mysql> insert into test (`id`, `name`) values (1, "hi~~~");
Query OK, 1 row affected (0.00 sec)
mysql> select * from test;
Empty set (0.00 sec)
mysql> commit;
Query OK, 0 rows affected (0.01 sec)
mysql> select * from test;
Empty set (0.00 sec)
mysql> insert into test (`id`, `name`) values (1, "hello");
ERROR 1062 (23000): Duplicate entry '1' for key 'PRIMARY'
//what fuck ???剛剛查詢,告訴我沒有數據。等我插入的時候就告訴我主鍵衝突了。此乃幻讀現象
複製代碼
session1: session2:
mysql> start transaction;
Query OK, 0 rows affected (0.01 sec)
mysql> start transaction;
Query OK, 0 rows affected (0.01 sec)
mysql> select * from test;
+----+-------+
| id | name |
+----+-------+
| 1 | hi~~~ |
+----+-------+
1 row in set (0.00 sec)
mysql> insert into test (`id`, `name`) values (2, "hello~~");
Query OK, 1 row affected (0.01 sec)
mysql> select * from test;
+----+-------+
| id | name |
+----+-------+
| 1 | hi~~~ |
+----+-------+
1 row in set (0.04 sec)
mysql> commit;
Query OK, 0 rows affected (0.00 sec)
mysql> select * from test;
+----+-------+
| id | name |
+----+-------+
| 1 | hi~~~ |
+----+-------+
1 row in set (0.00 sec)
mysql> update test set name = "up";
Query OK, 2 rows affected (0.01 sec)
Rows matched: 2 Changed: 2 Warnings: 0
//what fuck ???剛出查詢不是隻有一條數據嗎?怎麼更新了兩條。此乃幻讀現象
mysql> select * from test;
+----+------+
| id | name |
+----+------+
| 1 | up |
| 2 | up |
+----+------+
2 rows in set (0.00 sec)
複製代碼
session1: session2:
mysql> start transaction;
Query OK, 0 rows affected (0.00 sec)
mysql> start transaction;
Query OK, 0 rows affected (0.00 sec)
//加鎖鎖住了 id <= 1 的範圍
mysql> select * from test where id <= 1 for update;
+----+------+
| id | name |
+----+------+
| 1 | up |
+----+------+
1 row in set (0.18 sec)
//id 不在鎖內,容許插入
mysql> insert into test (`id`, `name`) values (3, "lock");
Query OK, 1 row affected (0.15 sec)
mysql> select * from test;
+----+------+
| id | name |
+----+------+
| 1 | up |
| 2 | up |
+----+------+
2 rows in set (0.01 sec)
//id = 1 已經加了寫鎖,事務等待鎖釋放
mysql> insert into test(`id`, `name`) values (1, "lock");
ERROR 1205 (HY000): Lock wait timeout exceeded; try restarting transaction
mysql> select * from test;
+----+------+
| id | name |
+----+------+
| 1 | up |
| 2 | up |
+----+------+
2 rows in set (0.00 sec)
mysql> commit;
Query OK, 0 rows affected (0.00 sec)
mysql> select * from test;
+----+------+
| id | name |
+----+------+
| 1 | up |
| 2 | up |
+----+------+
2 rows in set (0.00 sec)
mysql> commit;
Query OK, 0 rows affected (0.00 sec)
mysql> select * from test;
+----+------+
| id | name |
+----+------+
| 1 | up |
| 2 | up |
| 3 | lock | ------------session2 插入的數據
+----+------+
4 rows in set (0.00 sec)
複製代碼
session1: session2:
mysql> start transaction;
Query OK, 0 rows affected (0.00 sec)
mysql> start transaction;
Query OK, 0 rows affected (0.00 sec)
mysql> select * from test;
+----+------+
| id | name |
+----+------+
| 1 | up |
+----+------+
4 rows in set (0.01 sec)
mysql> insert into test (`id`, `name`) values (7, "hello");
Query OK, 1 row affected (0.00 sec)
mysql> commit;
Query OK, 0 rows affected (0.08 sec)
mysql> select * from test;
+----+------+
| id | name |
+----+------+
| 1 | up |
+----+------+
4 rows in set (0.00 sec)
mysql> select * from test;
+----+------+
| id | name |
+----+------+
| 1 | up |
+----+------+
4 rows in set (0.00 sec)
//讀到了 session2 提交的數據
mysql> select * from test lock in share mode;
+----+-------+
| id | name |
+----+-------+
| 1 | up |
| 7 | hello |
+----+-------+
5 rows in set (0.00 sec)
//讀到了 session2 提交的數據
mysql> select * from test for update;
+----+-------+
| id | name |
+----+-------+
| 1 | up |
| 7 | hello |
+----+-------+
5 rows in set (0.00 sec)
//讀不到 session2 提交的數據
mysql> select * from test;
+----+------+
| id | name |
+----+------+
| 1 | up |
+----+------+
4 rows in set (0.00 sec)
複製代碼