事務(wù)及事務(wù)隔離級(jí)別
什么是事務(wù)
事務(wù)是訪問(wèn)數(shù)據(jù)庫(kù)的一個(gè)操作序列,數(shù)據(jù)庫(kù)應(yīng)用系統(tǒng)通過(guò)事務(wù)集來(lái)完成對(duì)數(shù)據(jù)庫(kù)的存取帐我。事務(wù)的正確執(zhí)行使得數(shù)據(jù)庫(kù)從一種狀態(tài)轉(zhuǎn)換為另一種狀態(tài)。
事務(wù)必須服從ISO/IEC所制定的ACID原則阎毅。ACID是原子性(atomicity)焚刚、一致性(consistency)、隔離性(isolation)扇调、持久性(durability)的縮寫矿咕,這四種狀態(tài)的意思是:
1、原子性
即不可分割狼钮,事務(wù)要么全部被執(zhí)行碳柱,要么全部不執(zhí)行。如果事務(wù)的所有子事務(wù)全部提交成功熬芜,則所有的數(shù)據(jù)庫(kù)操作被提交莲镣,數(shù)據(jù)庫(kù)狀態(tài)發(fā)生變化;如果有子事務(wù)失敗涎拉,則其他子事務(wù)的數(shù)據(jù)庫(kù)操作被回滾瑞侮,即數(shù)據(jù)庫(kù)回到事務(wù)執(zhí)行前的狀態(tài),不會(huì)發(fā)生狀態(tài)轉(zhuǎn)換
2鼓拧、一致性
事務(wù)的執(zhí)行使得數(shù)據(jù)庫(kù)從一種正確狀態(tài)轉(zhuǎn)換成另外一種正確狀態(tài)
3半火、隔離性
在事務(wù)正確提交之前,不允許把事務(wù)對(duì)該數(shù)據(jù)的改變提供給任何其他事務(wù)季俩,即在事務(wù)正確提交之前钮糖,它可能的結(jié)果不應(yīng)該顯示給其他事務(wù)
4、持久性
事務(wù)正確提交之后酌住,其結(jié)果將永遠(yuǎn)保存在數(shù)據(jù)庫(kù)之中店归,即使在事務(wù)提交之后有了其他故障,事務(wù)的處理結(jié)果也會(huì)得到保存
事務(wù)的作用
事務(wù)管理對(duì)于企業(yè)級(jí)應(yīng)用而言至關(guān)重要酪我,它保證了用戶的每一次操作都是可靠的消痛,即便出現(xiàn)了異常的訪問(wèn)情況,也不至于破壞后臺(tái)數(shù)據(jù)的完整性都哭。就像銀行的自動(dòng)提款機(jī)ATM肄满,通常ATM都可以正常為客戶服務(wù)谴古,但是也難免遇到操作過(guò)程中及其突然出故障的情況,此時(shí)稠歉,事務(wù)就必須確保出故障前對(duì)賬戶的操作不生效,就像用戶剛才完全沒(méi)有使用過(guò)ATM機(jī)一樣汇陆,以保證用戶和銀行的利益都不受損失怒炸。
并發(fā)下事務(wù)會(huì)產(chǎn)生的問(wèn)題
舉個(gè)例子,事務(wù)A和事務(wù)B操縱的是同一個(gè)資源毡代,事務(wù)A有若干個(gè)子事務(wù)阅羹,事務(wù)B也有若干個(gè)子事務(wù),事務(wù)A和事務(wù)B在高并發(fā)的情況下教寂,會(huì)出現(xiàn)各種各樣的問(wèn)題捏鱼。"各種各樣的問(wèn)題",總結(jié)一下主要就是五種:第一類丟失更新酪耕、第二類丟失更新导梆、臟讀、不可重復(fù)讀迂烁、幻讀看尼。五種之中,第一類丟失更新盟步、第二類丟失更新不重要藏斩,不講了,講一下臟讀却盘、不可重復(fù)讀和幻讀狰域。
1、臟讀
所謂臟讀黄橘,就是指事務(wù)A讀到了事務(wù)B還沒(méi)有提交的數(shù)據(jù)兆览,比如銀行取錢,事務(wù)A開啟事務(wù)旬陡,此時(shí)切換到事務(wù)B拓颓,事務(wù)B開啟事務(wù)-->取走100元,此時(shí)切換回事務(wù)A描孟,事務(wù)A讀取的肯定是數(shù)據(jù)庫(kù)里面的原始數(shù)據(jù)驶睦,因?yàn)槭聞?wù)B取走了100塊錢,并沒(méi)有提交匿醒,數(shù)據(jù)庫(kù)里面的賬務(wù)余額肯定還是原始余額场航,這就是臟讀。
2廉羔、不可重復(fù)讀
所謂不可重復(fù)讀溉痢,就是指在一個(gè)事務(wù)里面讀取了兩次某個(gè)數(shù)據(jù),讀出來(lái)的數(shù)據(jù)不一致。還是以銀行取錢為例孩饼,事務(wù)A開啟事務(wù)-->查出銀行卡余額為1000元髓削,此時(shí)切換到事務(wù)B事務(wù)B開啟事務(wù)-->事務(wù)B取走100元-->提交,數(shù)據(jù)庫(kù)里面余額變?yōu)?00元镀娶,此時(shí)切換回事務(wù)A立膛,事務(wù)A再查一次查出賬戶余額為900元,這樣對(duì)事務(wù)A而言梯码,在同一個(gè)事務(wù)內(nèi)兩次讀取賬戶余額數(shù)據(jù)不一致宝泵,這就是不可重復(fù)讀。
3轩娶、幻讀
所謂幻讀儿奶,就是指在一個(gè)事務(wù)里面的操作中發(fā)現(xiàn)了未被操作的數(shù)據(jù)。比如學(xué)生信息鳄抒,事務(wù)A開啟事務(wù)-->修改所有學(xué)生當(dāng)天簽到狀況為false闯捎,此時(shí)切換到事務(wù)B,事務(wù)B開啟事務(wù)-->事務(wù)B插入了一條學(xué)生數(shù)據(jù)嘁酿,此時(shí)切換回事務(wù)A隙券,事務(wù)A提交的時(shí)候發(fā)現(xiàn)了一條自己沒(méi)有修改過(guò)的數(shù)據(jù),這就是幻讀闹司,就好像發(fā)生了幻覺(jué)一樣娱仔。幻讀出現(xiàn)的前提是并發(fā)的事務(wù)中有事務(wù)發(fā)生了插入游桩、刪除操作牲迫。
事務(wù)隔離級(jí)別
事務(wù)隔離級(jí)別,就是為了解決上面幾種問(wèn)題而誕生的借卧。為什么要有事務(wù)隔離級(jí)別盹憎,因?yàn)?strong>事務(wù)隔離級(jí)別越高,在并發(fā)下會(huì)產(chǎn)生的問(wèn)題就越少铐刘,但同時(shí)付出的性能消耗也將越大****陪每,因此很多時(shí)候必須在并發(fā)性和性能之間做一個(gè)權(quán)衡。所以設(shè)立了幾種事務(wù)隔離級(jí)別镰吵,以便讓不同的項(xiàng)目可以根據(jù)自己項(xiàng)目的并發(fā)情況選擇合適的事務(wù)隔離級(jí)別檩禾,對(duì)于在事務(wù)隔離級(jí)別之外會(huì)產(chǎn)生的并發(fā)問(wèn)題,在代碼中做補(bǔ)償疤祭。
事務(wù)隔離級(jí)別有4種盼产,但是像Spring會(huì)提供給用戶5種,來(lái)看一下:
1勺馆、DEFAULT
默認(rèn)隔離級(jí)別戏售,每種數(shù)據(jù)庫(kù)支持的事務(wù)隔離級(jí)別不一樣侨核,如果Spring配置事務(wù)時(shí)將isolation設(shè)置為這個(gè)值的話,那么將使用底層數(shù)據(jù)庫(kù)的默認(rèn)事務(wù)隔離級(jí)別灌灾。順便說(shuō)一句搓译,如果使用的MySQL,可以使用"select @tx_isolation"來(lái)查看默認(rèn)的事務(wù)隔離級(jí)別
2紧卒、READ_UNCOMMITTED
讀未提交侥衬,即能夠讀取到?jīng)]有被提交的數(shù)據(jù),所以很明顯這個(gè)級(jí)別的隔離機(jī)制無(wú)法解決臟讀跑芳、不可重復(fù)讀、幻讀中的任何一種直颅,因此很少使用
3博个、READ_COMMITED
讀已提交,即能夠讀到那些已經(jīng)提交的數(shù)據(jù)功偿,自然能夠防止臟讀盆佣,但是無(wú)法限制不可重復(fù)讀和幻讀
4、REPEATABLE_READ
重復(fù)讀取械荷,即在數(shù)據(jù)讀出來(lái)之后加鎖共耍,類似"select * from XXX for update",明確數(shù)據(jù)讀取出來(lái)就是為了更新用的吨瞎,所以要加一把鎖痹兜,防止別人修改它。REPEATABLE_READ的意思也類似颤诀,讀取了一條數(shù)據(jù)字旭,這個(gè)事務(wù)不結(jié)束,別的事務(wù)就不可以改這條記錄崖叫,這樣就解決了臟讀遗淳、不可重復(fù)讀的問(wèn)題,但是幻讀的問(wèn)題還是無(wú)法解決
5心傀、SERLALIZABLE
串行化屈暗,最高的事務(wù)隔離級(jí)別,不管多少事務(wù)脂男,挨個(gè)運(yùn)行完一個(gè)事務(wù)的所有子事務(wù)之后才可以執(zhí)行另外一個(gè)事務(wù)里面的所有子事務(wù)养叛,這樣就解決了臟讀、不可重復(fù)讀和幻讀的問(wèn)題了
網(wǎng)上專門有圖用表格的形式列出了事務(wù)隔離級(jí)別解決的并發(fā)問(wèn)題:
[站外圖片上傳中...(image-1b1a56-1542202038860)]
再必須強(qiáng)調(diào)一遍疆液,不是事務(wù)隔離級(jí)別設(shè)置得越高越好一铅,事務(wù)隔離級(jí)別設(shè)置得越高,意味著勢(shì)必要花手段去加鎖用以保證事務(wù)的正確性堕油,那么效率就要降低潘飘,因此實(shí)際開發(fā)中往往要在效率和并發(fā)正確性之間做一個(gè)取舍肮之,一般情況下會(huì)設(shè)置為READ_COMMITED,此時(shí)避免了臟讀卜录,并發(fā)性也還不錯(cuò)戈擒,之后再通過(guò)一些別的手段去解決不可重復(fù)讀和幻讀的問(wèn)題就好了。
事物隔離級(jí)別查看及修改
首先說(shuō)明一下MySQL查看和修改事務(wù)隔離級(jí)別的幾個(gè)命令:
查看事務(wù)隔離級(jí)別使用select @@tx_isolation
修改當(dāng)前會(huì)話事務(wù)隔離級(jí)別使用SET session TRANSACTION ISOLATION LEVEL Serializable;(參數(shù)可以為:Read uncommitted|Read committed|Repeatable read|Serializable)
修改全局事務(wù)隔離級(jí)別使用SET global TRANSACTION ISOLATION LEVEL Serializable;(參數(shù)可以為:Read uncommitted|Read committed|Repeatable read|Serializable)
修改了會(huì)話的事務(wù)隔離級(jí)別艰毒,比如MyBatis筐高,getSqlSession()的時(shí)候,只針對(duì)這一次拿到的Session有效丑瞧;比如CMD命令行柑土,只對(duì)這一次的窗口有效。
修改了全局的事務(wù)隔離級(jí)別绊汹,那么針對(duì)此后所有的會(huì)話有效稽屏,當(dāng)前已經(jīng)存在的會(huì)話不受影響。
關(guān)于MySQL事務(wù)隔離級(jí)別西乖,推薦大家一篇文章狐榔,很詳細(xì)地測(cè)試了四種事務(wù)隔離級(jí)別https://www.cnblogs.com/snsdzjlz320/p/5761387.html,相信大家讀了一定有所進(jìn)步获雕。
mysql :
set autocommit = 0
SET session TRANSACTION ISOLATION LEVEL Read uncommitted
SELECT * from t_user
INSERT INTO t_user (id, username, password) values (3, 'aaa', 'bbb')
COMMIT
ROLLBACK