mysql不支持python3吗_MySQL的4种事务隔离级别你还不清楚吗?
前言
現(xiàn)在想把數(shù)據(jù)庫這塊整理出來,盡量用最簡潔的語言描述出來,供新人參考。
首先創(chuàng)建一個表 account。創(chuàng)建表的過程略過(由于 InnoDB 存儲引擎支持事務(wù),所以將表的存儲引擎設(shè)置為 InnoDB)。表的結(jié)構(gòu)如下:
然后往表中插入兩條數(shù)據(jù),插入后結(jié)果如下:
為了說明問題,我們打開兩個控制臺分別進(jìn)行登錄來模擬兩個用戶(暫且成為用戶 A 和用戶 B 吧),并設(shè)置當(dāng)前 MySQL 會話的事務(wù)隔離級別。
一. read uncommitted(讀取未提交數(shù)據(jù))
具體用戶 A 的操作如下:
set session transaction isolation level read uncommitted; start transaction; select * from account;結(jié)果如下:
用戶 B 的操作如下:
set session transaction isolation level read uncommitted; start transaction; update account set account=account+200 where id = 1;隨后我們在 A 用戶中查詢數(shù)據(jù),結(jié)果如下:
結(jié)論一
我們將事務(wù)隔離級別設(shè)置為 read uncommitted,即便是事務(wù)沒有 commit,但是我們?nèi)匀荒茏x到未提交的數(shù)據(jù),這是所有隔離級別中最低的一種
那么這么做有什么問題嗎?
那就是我們在一個事務(wù)中可以隨隨便便讀取到其他事務(wù)未提交的數(shù)據(jù),這還是比較麻煩的,我們叫臟讀。我不知道這個名字是怎么起的,為了增強(qiáng)大家的印象,可以這么想,這個事務(wù)好輕浮啊,饑渴到連別人沒提交的東西都等不及,真臟,呸!
實際上我們的數(shù)據(jù)改變了嗎?
答案是否定的,因為只有事務(wù) commit 后才會更新到數(shù)據(jù)庫。
二. read committed(可以讀取其他事務(wù)提交的數(shù)據(jù))--- 大多數(shù)數(shù)據(jù)庫默認(rèn)的隔離級別
同樣的辦法,我們將用戶 B 所在的會話當(dāng)前事務(wù)隔離級別設(shè)置為 read commited。
在用戶 A 所在的會話中我們執(zhí)行下面操作:
update account set account=account-200 where id=1;
我們將 id=1 的用戶 account 減 200。然后查詢,發(fā)現(xiàn) id=1 的用戶 account 變?yōu)?800。
在 B 用戶所在的會話中查詢:
結(jié)果如下:
我們會發(fā)現(xiàn)數(shù)據(jù)并沒有變,還是 1000。
接著在會話 A 中我們將事務(wù)提交:
在會話 B 中查詢結(jié)果如下:
結(jié)論二:
當(dāng)我們將當(dāng)前會話的隔離級別設(shè)置為 read committed 的時候,當(dāng)前會話只能讀取到其他事務(wù)提交的數(shù)據(jù),未提交的數(shù)據(jù)讀不到。
那么這么做有什么問題嗎?
那就是我們在會話 B 同一個事務(wù)中,讀取到兩次不同的結(jié)果。這就造成了不可重復(fù)讀,就是兩次讀取的結(jié)果不同。這種現(xiàn)象叫不可重復(fù)讀。
三. repeatable read(可重讀)---MySQL 默認(rèn)的隔離級別
現(xiàn)在有個需求,就是老板說在同一個事務(wù)中查詢結(jié)果必須保持一致,如果你是數(shù)據(jù)庫,你會怎么做?數(shù)據(jù)庫是這么做的。
在會話 B 中我們當(dāng)前事務(wù)隔離級別為 repeatable read。具體操作如下:
set session transaction isolation level repeatable read; start transaction;接著在會話 B 中查詢數(shù)據(jù):
我們在 A 用戶所在會話中為表 account 添加一條數(shù)據(jù):
insert into account(id,account) value(3,1000); commit;然后我們查詢看數(shù)據(jù)插入是否成功:
回到 B 用戶所在的會話,我們查詢結(jié)果:
用戶 B 在他所在的會話中想插入一條新數(shù)據(jù) id=3,value=1000。來我們操作下:
什么?竟然插不進(jìn)去,說我數(shù)據(jù)重復(fù)?
用戶 B 當(dāng)然不服啊,因為查詢到數(shù)據(jù)只有兩條啊,為什么插入 id=3 說我數(shù)據(jù)重復(fù)了呢?
我再看一遍,莫非我眼花了?
試想一下,在實際中用戶 A 和用戶 B 肯定是相互隔離的,彼此不知道操作什么。用戶 B 碰到這種現(xiàn)象,肯定會炸毛的啊,明明不存在的數(shù)據(jù),插入?yún)s說主鍵 id=3 數(shù)據(jù)重復(fù)了。
結(jié)論三:
當(dāng)我們將當(dāng)前會話的隔離級別設(shè)置為 repeatable read 的時候,當(dāng)前會話可以重復(fù)讀,就是每次讀取的結(jié)果集都相同,而不管其他事務(wù)有沒有提交。
有什么問題嗎?
管他呢,老板的要求滿足了。要一個事務(wù)中讀取的數(shù)據(jù)一致(可重復(fù)讀)。我只能這么做啊,打腫臉裝胖子。數(shù)據(jù)已經(jīng)發(fā)生改變,但是我還是要保持一致。但是,出現(xiàn)了用戶 B 面對的問題,這種現(xiàn)象叫幻讀(記得當(dāng)時就在這個地方糾結(jié)好久,到底什么是幻讀啊)。
四. serializable(串行化)
同樣,我們將用戶 B 所在的會話的事務(wù)隔離級別設(shè)置為 serializable 并開啟事務(wù)。
set session transaction isolation level serializable; start transaction;在用戶 B 所在的會話中我們執(zhí)行下面操作:
select * from account;結(jié)果如下:
那我們這個時候在用戶 A 所在的會話中寫數(shù)據(jù)呢?
我們發(fā)現(xiàn)用戶 A 所在的會話陷入等待,如果超時(這個時間可以進(jìn)行配置),會出現(xiàn) Lock wait time out 提示:
如果在等待期間我們用戶 B 所在的會話事務(wù)提交,那么用戶 A 所在的事務(wù)的寫操作將提示操作成功。
結(jié)論四:
當(dāng)我們將當(dāng)前會話的隔離級別設(shè)置為 serializable 的時候,其他會話對該表的寫操作將被掛起。可以看到,這是隔離級別中最嚴(yán)格的,但是這樣做勢必對性能造成影響。所以在實際的選用上,我們要根據(jù)當(dāng)前具體的情況選用合適的。
最后
歡迎大家一起交流,喜歡文章記得關(guān)注我點個贊喲,感謝支持!
總結(jié)
以上是生活随笔為你收集整理的mysql不支持python3吗_MySQL的4种事务隔离级别你还不清楚吗?的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: php调取 zabbix实时数据_Lin
- 下一篇: torch 变量_python变量