spring事務(wù):
隔離級(jí)別
隔離級(jí)別是指若干個(gè)并發(fā)的事務(wù)之間的隔離程度,與我們開發(fā)時(shí)候主要相關(guān)的場(chǎng)景包括:臟讀取、重復(fù)讀、幻讀。
- DEFAULT:這是默認(rèn)值,表示使用底層數(shù)據(jù)庫(kù)的默認(rèn)隔離級(jí)別。對(duì)大部分?jǐn)?shù)據(jù)庫(kù)而言,通常這值就是: - READ_COMMITTED 。
- READ_UNCOMMITTED :該隔離級(jí)別表示一個(gè)事務(wù)可以讀取另一個(gè)事務(wù)修改但還沒有提交的數(shù)據(jù)。該級(jí)別不能防止臟讀和不可重復(fù)讀,因此很少使用該隔離級(jí)別。
- READ_COMMITTED :該隔離級(jí)別表示一個(gè)事務(wù)只能讀取另一個(gè)事務(wù)已經(jīng)提交的數(shù)據(jù)。該級(jí)別可以防止臟讀,這也是大多數(shù)情況下的推薦值。
- REPEATABLE_READ :該隔離級(jí)別表示一個(gè)事務(wù)在整個(gè)過(guò)程中可以多次重復(fù)執(zhí)行某個(gè)查詢,并且每次返回的記錄都相同。即使在多次查詢之間有新增的數(shù)據(jù)滿足該查詢,這些新增的記錄也會(huì)被忽略。該級(jí)別可以防止臟讀和不可重復(fù)讀。
- SERIALIZABLE :所有的事務(wù)依次逐個(gè)執(zhí)行,這樣事務(wù)之間就完全不可能產(chǎn)生干擾,也就是說(shuō),該級(jí)別可以防止臟讀、不可重復(fù)讀以及幻讀。但是這將嚴(yán)重影響程序的性能。通常情況下也不會(huì)用到該級(jí)別。
參考:#什么是臟讀,不可重復(fù)讀,幻讀
傳播行為:
所謂事務(wù)的傳播行為是指,如果在開始當(dāng)前事務(wù)之前,一個(gè)事務(wù)上下文已經(jīng)存在,此時(shí)有若干選項(xiàng)可以指定一個(gè)事務(wù)性方法的執(zhí)行行為。
- REQUIRED :如果當(dāng)前存在事務(wù),則加入該事務(wù);如果當(dāng)前沒有事務(wù),則創(chuàng)建一個(gè)新的事務(wù)。
- SUPPORTS :如果當(dāng)前存在事務(wù),則加入該事務(wù);如果當(dāng)前沒有事務(wù),則以非事務(wù)的方式繼續(xù)運(yùn)行。
- MANDATORY :如果當(dāng)前存在事務(wù),則加入該事務(wù);如果當(dāng)前沒有事務(wù),則拋出異常。
- REQUIRES_NEW :創(chuàng)建一個(gè)新的事務(wù),如果當(dāng)前存在事務(wù),則把當(dāng)前事務(wù)掛起。
- NOT_SUPPORTED :以非事務(wù)方式運(yùn)行,如果當(dāng)前存在事務(wù),則把當(dāng)前事務(wù)掛起。
- NEVER :以非事務(wù)方式運(yùn)行,如果當(dāng)前存在事務(wù),則拋出異常。
- NESTED :如果當(dāng)前存在事務(wù),則創(chuàng)建一個(gè)事務(wù)作為當(dāng)前事務(wù)的嵌套事務(wù)來(lái)運(yùn)行;如果當(dāng)前沒有事務(wù),則該取值等價(jià)于 REQUIRED 。注意:事務(wù)A在外面,事務(wù)B在里面,那么A事務(wù)回滾會(huì)導(dǎo)致B回滾,B事務(wù)對(duì)A事務(wù)沒有影響。
@Transactional(propagation = Propagation.NESTED)
public boolean publish(Article article){
Content content = new Content();
content.setContent(article.getContent());
for (int i = 0; i < 3; i++) {
//這里執(zhí)行過(guò)程沒有一行 且有事務(wù)控制
boolean insertOK = contentService.insert(content);
if (!insertOK) {
log.error("發(fā)布文章失敗:寫入文本內(nèi)容異常");
throw new RuntimeException("發(fā)布文章失敗");
}
}
article.setContentId(content.getId());
LocalDateTime now = LocalDateTime.now();
article.setUTime(now);
article.setUTime(now);
//拋出異常 導(dǎo)致publish()方法上的事務(wù)提交失敗 導(dǎo)致contentService.insert(content);回滾【即使后者執(zhí)行了多次也得回滾】
if(true) throw new RuntimeException();
return insert(article);
}
分布式事務(wù)
分布式事務(wù)處理 : Java 事務(wù)編程接口(JTA:Java Transaction API)和 Java 事務(wù)服務(wù) (JTS;Java Transaction Service) 為 J2EE 平臺(tái)提供了分布式事務(wù)服務(wù)。分布式事務(wù)(Distributed Transaction)包括事務(wù)管理器(Transaction Manager)和一個(gè)或多個(gè)支持 XA 協(xié)議的資源管理器 ( Resource Manager )。我們可以將資源管理器看做任意類型的持久化數(shù)據(jù)存儲(chǔ);事務(wù)管理器承擔(dān)著所有事務(wù)參與單元的協(xié)調(diào)與控制。
參考:JTA 深度歷險(xiǎn) - 原理與實(shí)現(xiàn)