javascript
Spring 事务相关及@Transactional的使用建议
使用步驟:
步驟一、在spring配置文件中引入<tx:>命名空間
<beans xmlns="http://www.springframework.org/schema/beans"
?xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
?xmlns:tx="http://www.springframework.org/schema/tx"
?xsi:schemaLocation="http://www.springframework.org/schema/beans
?http://www.springframework.org/schema/beans/spring-beans-2.0.xsd
?http://www.springframework.org/schema/tx
?http://www.springframework.org/schema/tx/spring-tx-2.0.xsd">
步驟二、具有@Transactional 注解的bean自動配置為聲明式事務(wù)支持?
?
步驟三、在接口或類的聲明處 ,寫一個@Transactional.
要是只在接口上寫, 接口的實現(xiàn)類就會繼承下來、接口的實現(xiàn)類的具體方法,可以覆蓋類聲明處的設(shè)置
@Transactional ? //類級的注解、適用于類中所有的public的方法
事務(wù)的傳播行為和隔離級別
大家在使用spring的注解式事務(wù)管理時,對事務(wù)的傳播行為和隔離級別可能有點不知所措,下邊就詳細的介紹下以備方便查閱。
事物注解方式: @Transactional
當標于類前時, 標示類中所有方法都進行事物處理?,?例子:
@Transactional public class TestServiceBean implements TestService {}當類中某些方法不需要事物時:
@Transactional public class TestServiceBean implements TestService { private TestDao dao; public void setDao(TestDao dao) {this.dao = dao;} @Transactional(propagation = Propagation.NOT_SUPPORTED)public List<Object> getAll() {return null;} }事物傳播行為介紹:?
@Transactional(propagation=Propagation.REQUIRED)?
如果有事務(wù), 那么加入事務(wù), 沒有的話新建一個(默認情況下)
@Transactional(propagation=Propagation.NOT_SUPPORTED)?
容器不為這個方法開啟事務(wù)
@Transactional(propagation=Propagation.REQUIRES_NEW)?
不管是否存在事務(wù),都創(chuàng)建一個新的事務(wù),原來的掛起,新的執(zhí)行完畢,繼續(xù)執(zhí)行老的事務(wù)
@Transactional(propagation=Propagation.MANDATORY)?
必須在一個已有的事務(wù)中執(zhí)行,否則拋出異常
@Transactional(propagation=Propagation.NEVER)?
必須在一個沒有的事務(wù)中執(zhí)行,否則拋出異常(與Propagation.MANDATORY相反)
@Transactional(propagation=Propagation.SUPPORTS)?
如果其他bean調(diào)用這個方法,在其他bean中聲明事務(wù),那就用事務(wù).如果其他bean沒有聲明事務(wù),那就不用事務(wù).
事物超時設(shè)置:
@Transactional(timeout=30) //默認是30秒
事務(wù)隔離級別:
@Transactional(isolation = Isolation.READ_UNCOMMITTED)
讀取未提交數(shù)據(jù)(會出現(xiàn)臟讀, 不可重復讀) 基本不使用
@Transactional(isolation = Isolation.READ_COMMITTED)
讀取已提交數(shù)據(jù)(會出現(xiàn)不可重復讀和幻讀)
@Transactional(isolation = Isolation.REPEATABLE_READ)
可重復讀(會出現(xiàn)幻讀)
@Transactional(isolation = Isolation.SERIALIZABLE)
串行化
MYSQL: 默認為REPEATABLE_READ級別
SQLSERVER: 默認為READ_COMMITTED
臟讀?: 一個事務(wù)讀取到另一事務(wù)未提交的更新數(shù)據(jù)
不可重復讀?: 在同一事務(wù)中, 多次讀取同一數(shù)據(jù)返回的結(jié)果有所不同, 換句話說,?
后續(xù)讀取可以讀到另一事務(wù)已提交的更新數(shù)據(jù). 相反, "可重復讀"在同一事務(wù)中多次
讀取數(shù)據(jù)時, 能夠保證所讀數(shù)據(jù)一樣, 也就是后續(xù)讀取不能讀到另一事務(wù)已提交的更新數(shù)據(jù)
幻讀?: 一個事務(wù)讀到另一個事務(wù)已提交的insert數(shù)據(jù)
@Transactional注解中常用參數(shù)說明
| 參 數(shù) 名 稱 | 功 能 描 述 |
| readOnly | 該屬性用于設(shè)置當前事務(wù)是否為只讀事務(wù),設(shè)置為true表示只讀,false則表示可讀寫,默認值為false。例如:@Transactional(readOnly=true) |
| rollbackFor | 該屬性用于設(shè)置需要進行回滾的異常類數(shù)組,當方法中拋出指定異常數(shù)組中的異常時,則進行事務(wù)回滾。例如: 指定單一異常類:@Transactional(rollbackFor=RuntimeException.class) 指定多個異常類:@Transactional(rollbackFor={RuntimeException.class, Exception.class}) |
?續(xù)表)
| 參 數(shù) 名 稱 | 功 能 描 述 |
| rollbackForClassName | 該屬性用于設(shè)置需要進行回滾的異常類名稱數(shù)組,當方法中拋出指定異常名稱數(shù)組中的異常時,則進行事務(wù)回滾。例如: 指定單一異常類名稱:@Transactional(rollbackForClassName="RuntimeException") 指定多個異常類名稱:@Transactional(rollbackForClassName={"RuntimeException","Exception"}) |
| noRollbackFor | 該屬性用于設(shè)置不需要進行回滾的異常類數(shù)組,當方法中拋出指定異常數(shù)組中的異常時,不進行事務(wù)回滾。例如: 指定單一異常類:@Transactional(noRollbackFor=RuntimeException.class) 指定多個異常類:@Transactional(noRollbackFor={RuntimeException.class, Exception.class}) |
| noRollbackForClassName | 該屬性用于設(shè)置不需要進行回滾的異常類名稱數(shù)組,當方法中拋出指定異常名稱數(shù)組中的異常時,不進行事務(wù)回滾。例如: 指定單一異常類名稱:@Transactional(noRollbackForClassName="RuntimeException") 指定多個異常類名稱: @Transactional(noRollbackForClassName={"RuntimeException","Exception"}) |
| propagation | 該屬性用于設(shè)置事務(wù)的傳播行為,具體取值可參考表6-7。 例如:@Transactional(propagation=Propagation.NOT_SUPPORTED,readOnly=true) |
| isolation | 該屬性用于設(shè)置底層數(shù)據(jù)庫的事務(wù)隔離級別,事務(wù)隔離級別用于處理多事務(wù)并發(fā)的情況,通常使用數(shù)據(jù)庫的默認隔離級別即可,基本不需要進行設(shè)置 |
| timeout | 該屬性用于設(shè)置事務(wù)的超時秒數(shù),默認值為-1表示永不超時 |
注意的幾點:
1 @Transactional 只能被應用到public方法上, 對于其它非public的方法,如果標記了@Transactional也不會報錯,但方法沒有事務(wù)功能.
2用 spring 事務(wù)管理器,由spring來負責數(shù)據(jù)庫的打開,提交,回滾.默認遇到運行期例外(throw new RuntimeException("注釋");)會回滾,即遇到不受檢查(unchecked)的例外時回滾;而遇到需要捕獲的例外(throw new Exception("注釋");)不會回滾,即遇到受檢查的例外(就是非運行時拋出的異常,編譯器會檢查到的異常叫受檢查例外或說受檢查異常)時,需我們指定方式來讓事務(wù)回滾 要想所有異常都回滾,要加上 @Transactional( rollbackFor={Exception.class,其它異常}) .如果讓unchecked例外不回滾: @Transactional(notRollbackFor=RunTimeException.class)
如下:
@Transactional(rollbackFor=Exception.class) //指定回滾,遇到異常Exception時回滾
public void methodName() {
throw new Exception("注釋");
}
@Transactional(noRollbackFor=Exception.class)//指定不回滾,遇到運行期例外(throw new RuntimeException("注釋");)會回滾
public ItimDaoImpl getItemDaoImpl() {
throw new RuntimeException("注釋");
}
3、@Transactional 注解應該只被應用到 public 可見度的方法上。 如果你在 protected、private 或者 package-visible 的方法上使用 @Transactional 注解,它也不會報錯, 但是這個被注解的方法將不會展示已配置的事務(wù)設(shè)置。
4、@Transactional 注解可以被應用于接口定義和接口方法、類定義和類的 public 方法上。然而,請注意僅僅 @Transactional 注解的出現(xiàn)不足于開啟事務(wù)行為,它僅僅 是一種元數(shù)據(jù),能夠被可以識別 @Transactional 注解和上述的配置適當?shù)木哂惺聞?wù)行為的beans所使用。上面的例子中,其實正是 <tx:annotation-driven/>元素的出現(xiàn) 開啟 了事務(wù)行為。
5、Spring團隊的建議是你在具體的類(或類的方法)上使用 @Transactional 注解,而不要使用在類所要實現(xiàn)的任何接口上。你當然可以在接口上使用 @Transactional 注解,但是這將只能當你設(shè)置了基于接口的代理時它才生效。因為注解是 不能繼承 的,這就意味著如果你正在使用基于類的代理時,那么事務(wù)的設(shè)置將不能被基于類的代理所識別,而且對象也將不會被事務(wù)代理所包裝(將被確認為嚴重的)。因 此,請接受Spring團隊的建議并且在具體的類上使用 @Transactional 注解。
轉(zhuǎn)載于:https://www.cnblogs.com/albertzhangyu/p/9693346.html
創(chuàng)作挑戰(zhàn)賽新人創(chuàng)作獎勵來咯,堅持創(chuàng)作打卡瓜分現(xiàn)金大獎總結(jié)
以上是生活随笔為你收集整理的Spring 事务相关及@Transactional的使用建议的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 制定项目管理计划
- 下一篇: 面试一口气说出Spring的声明式事务@