👨?🎓作者簡介:一位大四、研0學生,正在努力準備大四暑假的實習、
🌌上期文章:MySQL進階:視圖&&存儲過程&&存儲函數&&觸發器
📚訂閱專欄:MySQL進階
希望文章對你們有所幫助
這一部分內容還是很重要的,雖然學起來感覺不是很難,但是要多體會一下,體會好這里的精髓才能更好的理解MySQL事務的鎖。
MySQL事務及隔離級別
- 簡介
- 事務操作
- 未控制事務
- 控制事務一
- 控制事務二
- 事務四大特性(高頻面試題)
- 事務并發問題(重點)
- 事務隔離級別
簡介
事務是一組操作的集合,它是一個不可分割的工作單位,事務會把所有的操作作為一個整體一起向系統提交或撤銷操作請求,即這些操作要么同時成功,要么同時失敗。
就比如轉賬,包括加錢和減錢的操作,這一組操作必須在一個事務的范圍內,要么都成功,要么都失敗。
默認MySQL的事務是自動提交的,也就是說,當執行完一條DML語句時,MySQL會立即隱
式的提交事務。
事務操作
create table account(id int primary key AUTO_INCREMENT comment 'ID',name varchar(10) comment '姓名',money double(10,2) comment '余額'
) comment '賬戶表';
insert into account(name, money) VALUES ('張三',2000), ('李四',2000);
未控制事務
-- 1. 查詢張三余額
select * from account where name = '張三';
-- 2. 張三的余額減少1000
update account set money = money - 1000 where name = '張三';
出錯了....
-- 3. 李四的余額增加1000
update account set money = money + 1000 where name = '李四';
"出錯了…"這句話會報錯,若不加事務,就會導致張三的錢減少,但是李四的錢沒有增加,這就違背了數據一致性
。
控制事務一
1、查看/設置事務提交方式
SELECT @@autocommit ;
SET @@autocommit = 0 ;
2、提交事務
COMMIT;
3、回滾事務
ROLLBACK;
上述的這種方式,是修改了事務的自動提交行為, 把默認的自動提交修改為了手動提交, 此時我們執行的DML語句都不會提交, 需要手動的執行commit進行提交。
控制事務二
1、開啟事務
BEGIN; 或 START TRANSACTION;
2、提交事務
COMMIT;
3、回滾事務
ROLLBACK;
事務四大特性(高頻面試題)
- 原子性:事務是不可分割的最小單元,要么全成功要么全失敗
- 一致性:事務完場時,必須使所有數據保持一致狀態
- 隔離性:數據庫系統提供的隔離機制,保證事務在不受外部并發操作影響的獨立環境下運行
- 持久性:事務一旦提交或回滾,對數據庫中的數據的改變是永久的
事務的四大特性,簡稱ACID。
事務并發問題(重點)
1、臟讀:一個事務讀到另外一個事務還沒有提交的數據。
B事務的select操作讀取到了DB中id=1的數據,但讀出來的是事務A執行update后的數據,但事務A根本還沒有提交事務。
2、不可重復讀:一個事務先后讀取同一條記錄,但兩次讀取的數據不同,稱之為不可重復讀。
注意:假設此時已經解決了事務的臟讀問題
事務A兩次讀取同一條記錄,但是讀取到的數據卻是不一樣的。這是因為在兩次select操作之間有一個事務B提交了,事務就是將這一行的數據修改了。
3、幻讀:一個事務按照條件查詢數據時,沒有對應數據行,但是在插入數據時,又發現這行數據已經存在,好像出現了“幻影”。
注意:假設此時已經解決了事務的臟讀問題和不可重復讀問題
1、事務A執行select,確認了數據庫沒有id=1的數據
2、事務B執行insert,數據庫中增加了id=1的數據
3、事務A執行insert,由于id=1的數據已經存在,無法插入
4、事務A再次執行select,由于已經解決了不可重復讀的問題,此時事務A還是沒辦法獲取到id=1的數據
這就是幻讀,明明查不到,卻插入不進去。
事務隔離級別
為了解決并發事務問題,數據庫中引入了事務的隔離級別:
隔離級別 | 臟讀 | 不可重復讀 | 幻讀 |
---|---|---|---|
Read uncommitted | √ | √ | √ |
Read commited | × | √ | √ |
Repeatable Read(默認) | × | × | √ |
Serializable | × | × | × |
1、查看事務的隔離級別:
SELECT @@TRANSACTION_ISOLATION;
2、設置事務隔離級別:
SET [ SESSION | GLOBAL ] TRANSACTION ISOLATION LEVEL { READ UNCOMMITTED |
READ COMMITTED | REPEATABLE READ | SERIALIZABLE }
需要注意的是,事務的隔離級別越高,數據自然越安全,但是性能會更低。
在這里我們可以看到,MySQL默認的事務隔離級別是RR,臟讀和不可重復讀的問題是已經解決了,但幻讀問題的解決方法是要加一個行級鎖——間隙鎖&臨鍵鎖。