前言
數(shù)據(jù)庫(kù)應(yīng)該是所有后端開(kāi)發(fā)都會(huì)涉及的移袍,數(shù)據(jù)庫(kù)事務(wù),很多業(yè)務(wù)場(chǎng)景可能并不要求很高,而且也很少出現(xiàn)进倍,經(jīng)常出現(xiàn)在高并發(fā)土至,高流量的情況下,在未來(lái)迎接挑戰(zhàn)的時(shí)候猾昆,需要掌握相關(guān)的知識(shí)陶因,這里希望能夠簡(jiǎn)單介紹下,總結(jié)一點(diǎn)知識(shí)點(diǎn)和注意點(diǎn)垂蜗。
事務(wù)
目前接觸到的最多的就是通過(guò)spring來(lái)控制事務(wù)處理楷扬。這里就重點(diǎn)講一下spring事務(wù)的知識(shí)點(diǎn)。
事務(wù)的目的:
- 數(shù)據(jù)一致:指的是事務(wù)提交能夠正確執(zhí)行么抗,而且持久到數(shù)據(jù)庫(kù)毅否,事務(wù)回滾能夠恢復(fù)到事務(wù)執(zhí)行之前的狀態(tài)。
- 操作隔離:多個(gè)事務(wù)操作應(yīng)該相互獨(dú)立蝇刀,不干擾螟加。
這里就引出了事務(wù)的四個(gè)特性:
- 原子性 (atomicity):強(qiáng)調(diào)事務(wù)的不可分割.
- 一致性 (consistency):事務(wù)的執(zhí)行的前后數(shù)據(jù)的完整性保持一致.
- 隔離性 (isolation):一個(gè)事務(wù)執(zhí)行的過(guò)程中,不應(yīng)該受到其他事務(wù)的干擾
- 持久性(durability) :事務(wù)一旦結(jié)束,數(shù)據(jù)就持久到數(shù)據(jù)庫(kù)
我們把嚴(yán)格遵循ACID
特性的事務(wù)成為剛性事務(wù)
。
期望最終一致性吞琐,在事務(wù)執(zhí)行的中間狀態(tài)允許暫時(shí)不遵循ACID
特性的事務(wù)稱為柔性事務(wù)
捆探。
事務(wù)常見(jiàn)異常
對(duì)于事務(wù)異常,總的可以分為三類:
- 臟讀:當(dāng)前有事務(wù)在運(yùn)行站粟,另一個(gè)線程獲取到數(shù)據(jù)黍图,前面的事務(wù)可能回滾,所以出現(xiàn)了臟讀奴烙,這個(gè)現(xiàn)象在事務(wù)隔離級(jí)別
Read uncommitted
時(shí)會(huì)出現(xiàn)杯巨。 - 不可重復(fù)讀:事務(wù)a對(duì)數(shù)據(jù)讀取了兩次娃属,再此過(guò)程中跋涣,事務(wù)B對(duì)數(shù)據(jù)進(jìn)行了修改斜纪,這時(shí),事務(wù)a前后兩次數(shù)據(jù)讀取就不一致了幅虑,當(dāng)事務(wù)隔離級(jí)別設(shè)置為
Read uncommitted
丰滑、Read committed
時(shí)會(huì)出現(xiàn)。 - 幻讀:事務(wù)a對(duì)數(shù)據(jù)讀取兩次倒庵,在此過(guò)程中褒墨,事務(wù)B對(duì)數(shù)據(jù)進(jìn)行了新增和刪除,這時(shí)擎宝,事務(wù)a前后兩次獲取的數(shù)據(jù)量就不一致了郁妈。當(dāng)事務(wù)隔離級(jí)別設(shè)置為
Read uncommitted
、Read committed
绍申、Repeatable read
針對(duì)事務(wù)常見(jiàn)異常圃庭,我們可知,數(shù)據(jù)庫(kù)可以設(shè)置對(duì)應(yīng)的事務(wù)隔離級(jí)別,針對(duì)不同的業(yè)務(wù)進(jìn)行不同的配置剧腻。
- Read uncommitted:設(shè)置該級(jí)別,事務(wù)b可以讀取事務(wù)a未提交的事務(wù)涂屁。
后續(xù)可以跟進(jìn)下具體的原理书在。
- Read committed:設(shè)置該級(jí)別,事務(wù)b只能獲取事務(wù)a已經(jīng)提交的事務(wù)拆又,但是不能保證事務(wù)b在事務(wù)a提交前后數(shù)據(jù)保持一致儒旬。
- Repeatable read:設(shè)置該級(jí)別,事務(wù)a如果已經(jīng)獲取了某條數(shù)據(jù)帖族,事務(wù)b在事務(wù)a完成之前栈源,是不允許修改該數(shù)據(jù)的,但是可以修改其他數(shù)據(jù)竖般,這就導(dǎo)致了幻讀甚垦。
注:Mysql的默認(rèn)隔離級(jí)別就是Repeatable read。 - Serializable:所有事務(wù)順序執(zhí)行涣雕。
事務(wù)死鎖
在高并發(fā)的情況下艰亮,為了讓數(shù)據(jù)更可靠,不可避免的會(huì)進(jìn)行事務(wù)挣郭,事務(wù)會(huì)對(duì)表和數(shù)據(jù)進(jìn)行加鎖迄埃。這也就要考慮事務(wù)死鎖的問(wèn)題。
常見(jiàn)的鎖從不同的分類可以歸類成:
- 行鎖兑障、表鎖
- 共享鎖侄非、排它鎖(獨(dú)占鎖)
后續(xù)添加sql展示
具體使用場(chǎng)景待分析
事務(wù)死鎖場(chǎng)景舉例:
- 1.轉(zhuǎn)賬問(wèn)題:
a和b互相轉(zhuǎn)賬,(a)a給b轉(zhuǎn)賬流译,a扣錢(qián)逞怨,鎖定a數(shù)據(jù),嘗試獲取b加錢(qián)的事務(wù)先蒋,(b)b給a轉(zhuǎn)賬骇钦,b扣錢(qián),鎖定b數(shù)據(jù)竞漾,嘗試獲取a加錢(qián)的事務(wù)眯搭。
解決方案:(1)按照賬號(hào)順序,就是說(shuō)业岁,始終a先處理事務(wù)鳞仙。(2)提前獲取a和b的事務(wù)。
Spring事務(wù)
Spring使用AOP(面向切面編程)來(lái)實(shí)現(xiàn)聲明式事務(wù)笔时,也可以使用編程式事務(wù)來(lái)控制事務(wù)棍好,需要了解動(dòng)態(tài)代理
和AOP增強(qiáng)
。
關(guān)于AOP增強(qiáng)
可以適當(dāng)了解下,主要是五種增強(qiáng)方式
借笙。
Spring事務(wù)抽象
- PlatformTransactionManager 事務(wù)管理器扒怖,聽(tīng)名字就知道它是管理事務(wù)的操作的,它只包含三個(gè)方法业稼。獲取事務(wù)盗痒,回顧事務(wù),提交事務(wù)
- TransactionDefiition 定義事務(wù)的類型低散,事務(wù)包含很多屬性俯邓,是否可讀,事務(wù)隔離級(jí)別,事務(wù)傳播級(jí)別熔号。通過(guò)事務(wù)的定義稽鞭,我們根據(jù)定義獲取特定的事務(wù)。
- TransactionStatus 代表一個(gè)事務(wù)運(yùn)行的狀態(tài)引镊,事務(wù)管理器通過(guò)狀態(tài)可以知道事務(wù)的狀態(tài)信息朦蕴,然后進(jìn)行事務(wù)的控制。事務(wù)是否完成祠乃,是否是新的事務(wù)梦重,是不是只能回滾等。
Spring事務(wù)攔截
聲明式事務(wù)是一種環(huán)繞增強(qiáng)亮瓷,對(duì)應(yīng)接口為MethodInterceptor琴拧,事務(wù)增強(qiáng)對(duì)該接口的實(shí)現(xiàn)為T(mén)ransactionInterceptor,查看父類接口可以看到對(duì)應(yīng)聲明式事務(wù)的提交嘱支,當(dāng)然這里的提交也可以用編程式事務(wù)處理蚓胸。
Transaction對(duì)象是由TransactionSynchronizationManager來(lái)控制的。其判斷當(dāng)前的數(shù)據(jù)源是否有連接對(duì)應(yīng)除师,用的是ThreadLocal沛膳,所以,事務(wù)是單線程的
Spring五個(gè)事務(wù)隔離級(jí)別和七個(gè)事務(wù)傳播行為
ISOLATION_DEFAULT 這是一個(gè)PlatfromTransactionManager默認(rèn)的隔離級(jí)別汛聚,使用數(shù)據(jù)庫(kù)默認(rèn)的事務(wù)隔離級(jí)別.另外四個(gè)與JDBC的隔離級(jí)別相對(duì)應(yīng)
ISOLATION_READ_UNCOMMITTED 這是事務(wù)最低的隔離級(jí)別锹安,它充許別外一個(gè)事務(wù)可以看到這個(gè)事務(wù)未提交的數(shù)據(jù)。這種隔離級(jí)別會(huì)產(chǎn)生臟讀倚舀,不可重復(fù)讀和幻像讀
ISOLATION_READ_COMMITTED 保證一個(gè)事務(wù)修改的數(shù)據(jù)提交后才能被另外一個(gè)事務(wù)讀取叹哭。另外一個(gè)事務(wù)不能讀取該事務(wù)未提交的數(shù)據(jù)。這種事務(wù)隔離級(jí)別可以避免臟讀出現(xiàn)痕貌,但是可能會(huì)出現(xiàn)不可重復(fù)讀和幻像讀风罩。
ISOLATION_REPEATABLE_READ 這種事務(wù)隔離級(jí)別可以防止臟讀,不可重復(fù)讀舵稠。但是可能出現(xiàn)幻像讀超升。它除了保證一個(gè)事務(wù)不能讀取另一個(gè)事務(wù)未提交的數(shù)據(jù)外入宦,還保證了避免下面的情況產(chǎn)生(不可重復(fù)讀)。
ISOLATION_SERIALIZABLE 這是花費(fèi)最高代價(jià)但是最可靠的事務(wù)隔離級(jí)別室琢。事務(wù)被處理為順序執(zhí)行乾闰。除了防止臟讀,不可重復(fù)讀外盈滴,還避免了幻像讀汹忠。
PROPAGATION_REQUIRED 如果存在一個(gè)事務(wù),則支持當(dāng)前事務(wù)雹熬。如果沒(méi)有事務(wù)則開(kāi)啟一個(gè)新的事務(wù)。
PROPAGATION_SUPPORTS 如果存在一個(gè)事務(wù)谣膳,支持當(dāng)前事務(wù)竿报。如果沒(méi)有事務(wù),則非事務(wù)的執(zhí)行继谚。但是對(duì)于事務(wù)同步的事務(wù)管理器烈菌,PROPAGATION_SUPPORTS與不使用事務(wù)有少許不同。
PROPAGATION_MANDATORY 如果已經(jīng)存在一個(gè)事務(wù)花履,支持當(dāng)前事務(wù)芽世。如果沒(méi)有一個(gè)活動(dòng)的事務(wù),則拋出異常诡壁。
PROPAGATION_REQUIRES_NEW 總是開(kāi)啟一個(gè)新的事務(wù)济瓢。如果一個(gè)事務(wù)已經(jīng)存在,則將這個(gè)存在的事務(wù)掛起妹卿。
PROPAGATION_NOT_SUPPORTED 總是非事務(wù)地執(zhí)行旺矾,并掛起任何存在的事務(wù)。
PROPAGATION_NEVER 總是非事務(wù)地執(zhí)行夺克,如果存在一個(gè)活動(dòng)事務(wù)箕宙,則拋出異常
PROPAGATION_NESTED如果一個(gè)活動(dòng)的事務(wù)存在,則運(yùn)行在一個(gè)嵌套的事務(wù)中. 如果沒(méi)有活動(dòng)事務(wù), 則按TransactionDefinition.PROPAGATION_REQUIRED 屬性執(zhí)行
待補(bǔ)充示例:
單線程事務(wù)
多線程事務(wù)
使用countdownlatch完成
分布式事務(wù)
解決方案分析
mq