事務(wù)的隔離級(jí)別
成都網(wǎng)絡(luò)公司-成都網(wǎng)站建設(shè)公司創(chuàng)新互聯(lián)建站十余年經(jīng)驗(yàn)成就非凡,專業(yè)從事成都網(wǎng)站建設(shè)、網(wǎng)站制作,成都網(wǎng)頁設(shè)計(jì),成都網(wǎng)頁制作,軟文發(fā)布平臺(tái),廣告投放等。十余年來已成功提供全面的成都網(wǎng)站建設(shè)方案,打造行業(yè)特色的成都網(wǎng)站建設(shè)案例,建站熱線:13518219792,我們期待您的來電!
數(shù)據(jù)庫事務(wù)的隔離級(jí)別有4個(gè),由低到高依次為Read uncommitted、Read committed、Repeatable read、Serializable,這四個(gè)級(jí)別可以逐個(gè)解決臟讀、不可重復(fù)讀、幻讀這幾類問題。
√: 可能出現(xiàn) ×: 不會(huì)出現(xiàn)
臟讀不可重復(fù)讀幻讀
Read uncommitted√√√
Read committed×√√
Repeatable read××√
Serializable×××
注意:我們討論隔離級(jí)別的場景,主要是在多個(gè)事務(wù)并發(fā)的情況下,因此,接下來的講解都圍繞事務(wù)并發(fā)。
Read uncommitted 讀未提交
公司發(fā)工資了,領(lǐng)導(dǎo)把5000元打到singo的賬號(hào)上,但是該事務(wù)并未提交,而 singo正好去查看賬戶,發(fā)現(xiàn)工資已經(jīng)到賬,是5000元整,非常高興??墒遣恍业氖牵I(lǐng)導(dǎo)發(fā)現(xiàn)發(fā)給singo的工資金額不對(duì),是2000元,于是迅速 回滾了事務(wù),修改金額后,將事務(wù)提交,最后singo實(shí)際的工資只有2000元,singo空歡喜一場。
出現(xiàn)上述情況,即我們所說的臟讀,兩個(gè)并發(fā)的事務(wù),“事務(wù)A:領(lǐng)導(dǎo)給singo發(fā)工資”、“事務(wù)B:singo查詢工資賬戶”,事務(wù)B讀取了事務(wù)A尚未提交的數(shù)據(jù)。
當(dāng)隔離級(jí)別設(shè)置為Read uncommitted時(shí),就可能出現(xiàn)臟讀,如何避免臟讀,請(qǐng)看下一個(gè)隔離級(jí)別。
Read committed 讀提交
singo拿著工資卡去消費(fèi),系統(tǒng)讀取到卡里確實(shí)有2000元,而此時(shí)她的老婆也正好 在網(wǎng)上轉(zhuǎn)賬,把singo工資卡的2000元轉(zhuǎn)到另一賬戶,并在singo之前提交了事務(wù),當(dāng)singo扣款時(shí),系統(tǒng)檢查到singo的工資卡已經(jīng)沒有 錢,扣款失敗,singo十分納悶,明明卡里有錢,為何......
出現(xiàn)上述情況,即我們所說的不可重復(fù)讀,兩個(gè)并發(fā)的事務(wù),“事務(wù)A:singo消費(fèi)”、“事務(wù)B:singo的老婆網(wǎng)上轉(zhuǎn)賬”,事務(wù)A事先讀取了數(shù)據(jù),事務(wù)B緊接了更新了數(shù)據(jù),并提交了事務(wù),而事務(wù)A再次讀取該數(shù)據(jù)時(shí),數(shù)據(jù)已經(jīng)發(fā)生了改變。
當(dāng)隔離級(jí)別設(shè)置為Read committed時(shí),避免了臟讀,但是可能會(huì)造成不可重復(fù)讀。
大多數(shù)數(shù)據(jù)庫的默認(rèn)級(jí)別就是Read committed,比如Sql Server , Oracle。如何解決不可重復(fù)讀這一問題,請(qǐng)看下一個(gè)隔離級(jí)別。
Repeatable read 重復(fù)讀
當(dāng)隔離級(jí)別設(shè)置為Repeatable read時(shí),可以避免不可重復(fù)讀。當(dāng)singo拿著工資卡去消費(fèi)時(shí),一旦系統(tǒng)開始讀取工資卡信息(即事務(wù)開始),singo的老婆就不可能對(duì)該記錄進(jìn)行修改,也就是singo的老婆不能在此時(shí)轉(zhuǎn)賬。
雖然Repeatable read避免了不可重復(fù)讀,但還有可能出現(xiàn)幻讀。
singo的老婆工作在銀行部門,她時(shí)常通過銀行內(nèi)部系統(tǒng)查看singo的信用卡消費(fèi) 記錄。有一天,她正在查詢到singo當(dāng)月信用卡的總消費(fèi)金額(select sum(amount) from transaction where month = 本月)為80元,而singo此時(shí)正好在外面胡吃海塞后在收銀臺(tái)買單,消費(fèi)1000元,即新增了一條1000元的消費(fèi)記錄(insert transaction ... ),并提交了事務(wù),隨后singo的老婆將singo當(dāng)月信用卡消費(fèi)的明細(xì)打印到A4紙上,卻發(fā)現(xiàn)消費(fèi)總額為1080元,singo的老婆很詫異,以為出 現(xiàn)了幻覺,幻讀就這樣產(chǎn)生了。
注:Mysql的默認(rèn)隔離級(jí)別就是Repeatable read。
Serializable 序列化
Serializable是最高的事務(wù)隔離級(jí)別,同時(shí)代價(jià)也花費(fèi)最高,性能很低,一般很少使用,在該級(jí)別下,事務(wù)順序執(zhí)行,不僅可以避免臟讀、不可重復(fù)讀,還避免了幻像讀。
事務(wù)隔離的四個(gè)級(jí)別是未提交讀(Read Uncommitted)、提交讀(Read Committed)、可重復(fù)讀(Repeable Read)、可串行化(Serializable)。
1、未提交讀(Read Uncommitted):事務(wù)可以讀取未提交的數(shù)據(jù),也稱作臟讀(Dirty Read)。一般很少使用。
2、提交讀(Read Committed):是大都是DBMS(如:Oracle,SQLServer)默認(rèn)事務(wù)隔離。執(zhí)行兩次同意的查詢卻有不同的結(jié)果,也叫不可重復(fù)讀。
3、可重復(fù)讀(Repeable Read):是MySQL默認(rèn)事務(wù)隔離級(jí)別。能確保同一事務(wù)多次讀取同一數(shù)據(jù)的結(jié)果是一致的??梢越鉀Q臟讀的問題,但理論上無法解決幻讀(Phantom Read)的問題。
4、可串行化(Serializable):是最高的隔離級(jí)別。強(qiáng)制事務(wù)串行執(zhí)行,會(huì)在讀取的每一行數(shù)據(jù)上加鎖,這樣雖然能避免幻讀的問題,但也可能導(dǎo)致大量的超時(shí)和鎖爭用的問題。很少會(huì)應(yīng)用到這種級(jí)別,只有在非常需要確保數(shù)據(jù)的一致性且可以接受沒有并發(fā)的應(yīng)用場景下才會(huì)考慮。
事務(wù)隔離級(jí)別特點(diǎn)比較
從事務(wù)隔離級(jí)別的定義上可以看出,Serializable級(jí)別隔離性最高,但是其效率也最低,因?yàn)槠湟笏胁僮飨嗤涗浀氖聞?wù)都串行的執(zhí)行。
對(duì)于MySql而言,其默認(rèn)事務(wù)級(jí)別是Repeatable read,雖然在定義上講,這種隔離級(jí)別無法解決幻讀的問題,但是MySql使用了一種Next key-lock的算法來實(shí)現(xiàn)Repeatable read,這種算法是能夠解決幻讀問題的。
關(guān)于Next key-lock算法,在進(jìn)行查詢時(shí),其不僅會(huì)將當(dāng)前的操作記錄鎖住,也會(huì)將查詢所涉及到的范圍鎖住。
也就是說,其他事務(wù)如果想要在當(dāng)前事務(wù)查詢的范圍內(nèi)進(jìn)行數(shù)據(jù)操作,那么其是會(huì)被阻塞的,因而MySql在Repeatable read隔離級(jí)別下就已經(jīng)具備了Serializable隔離級(jí)別的事務(wù)隔離性。
以上內(nèi)容參考:百度百科-隔離級(jí)別
1.查看當(dāng)前會(huì)話隔離級(jí)別
select @@tx_isolation;
2.查看系統(tǒng)當(dāng)前隔離級(jí)別
select @@global.tx_isolation;
3.設(shè)置當(dāng)前會(huì)話隔離級(jí)別
set session transaction isolatin level repeatable read;
4.設(shè)置系統(tǒng)當(dāng)前隔離級(jí)別
set global transaction isolation level repeatable read;
5.命令行,開始事務(wù)時(shí)
set autocommit=off 或者 start transaction
關(guān)于隔離級(jí)別的理解
1.read uncommitted
可以看到未提交的數(shù)據(jù)(臟讀),舉個(gè)例子:別人說的話你都相信了,但是可能他只是說說,并不實(shí)際做。
2.read committed
讀取提交的數(shù)據(jù)。但是,可能多次讀取的數(shù)據(jù)結(jié)果不一致(不可重復(fù)讀,幻讀)。用讀寫的觀點(diǎn)就是:讀取的行數(shù)據(jù),可以寫。
3.repeatable read(MySQL默認(rèn)隔離級(jí)別)
可以重復(fù)讀取,但有幻讀。讀寫觀點(diǎn):讀取的數(shù)據(jù)行不可寫,但是可以往表中新增數(shù)據(jù)。在MySQL中,其他事務(wù)新增的數(shù)據(jù),看不到,不會(huì)產(chǎn)生幻讀。采用多版本并發(fā)控制(MVCC)機(jī)制解決幻讀問題。
4.serializable
可讀,不可寫。像java中的鎖,寫數(shù)據(jù)必須等待另一個(gè)事務(wù)結(jié)束。
首選你需要了解一下 數(shù)據(jù)臟讀,幻讀,等一些概念,其次是你要了解一下鎖這個(gè)概念,當(dāng)一條數(shù)據(jù)被讀取時(shí),處于鎖狀態(tài),其他的用戶無法對(duì)其進(jìn)行操作。