javascript
Spring事务处理,以及Spring事务的传播属性和隔离级别
本文轉(zhuǎn)自:http://www.cnblogs.com/younggun/archive/2013/07/16/3193800.html
事務(wù)的隔離級別:事務(wù)隔離級別用于處理多事務(wù)并發(fā)的情況,通常使用數(shù)據(jù)庫的默認(rèn)隔離級別即可,基本不需要進(jìn)行設(shè)置
使用步驟:
步驟一、在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ù)支持?
? ?<bean id="defaultTransactionManager" class="org.springframework.orm.hibernate3.HibernateTransactionManager">
? ? ? ?<property name="sessionFactory" ref="sessionFactory" />
? ?</bean>
? ?
? ?<!-- 使用annotation定義事務(wù) -->
? ?<tx:annotation-driven transaction-manager="defaultTransactionManager" proxy-target-class="true" />
<!--?事務(wù)管理器配置,?Hibernate單數(shù)據(jù)源事務(wù)?--><bean?id="defaultTransactionManager"?class="org.springframework.orm.hibernate3.HibernateTransactionManager"><property?name="sessionFactory"?ref="sessionFactory"?/></bean><!--?使用annotation定義事務(wù)?--><tx:annotation-driven?transaction-manager="defaultTransactionManager"?proxy-target-class="true"?/>
步驟三、在接口或類的聲明處 ,寫一個@Transactional.
要是只在接口上寫, 接口的實(shí)現(xiàn)類就會繼承下來、接口的實(shí)現(xiàn)類的具體方法,可以覆蓋類聲明處的設(shè)置
@Transactional ? //類級的注解、適用于類中所有的public的方法
事務(wù)的傳播行為和隔離級別
大家在使用spring的注解式事務(wù)管理時,對事務(wù)的傳播行為和隔離級別可能有點(diǎn)不知所措,下邊就詳細(xì)的介紹下以備方便查閱。
事物注解方式: @Transactional
當(dāng)標(biāo)于類前時, 標(biāo)示類中所有方法都進(jìn)行事物處理?,?例子:
@Transactionalpublic class TestServiceBean implements TestService {}當(dāng)類中某些方法不需要事物時:
@Transactionalpublic 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ù), 沒有的話新建一個(默認(rèn)情況下)
@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) //默認(rèn)是30秒
事務(wù)隔離級別:
@Transactional(isolation = Isolation.READ_UNCOMMITTED)
讀取未提交數(shù)據(jù)(會出現(xiàn)臟讀, 不可重復(fù)讀) 基本不使用
@Transactional(isolation = Isolation.READ_COMMITTED)
讀取已提交數(shù)據(jù)(會出現(xiàn)不可重復(fù)讀和幻讀)
@Transactional(isolation = Isolation.REPEATABLE_READ)
可重復(fù)讀(會出現(xiàn)幻讀)
@Transactional(isolation = Isolation.SERIALIZABLE)
串行化
MYSQL: 默認(rèn)為REPEATABLE_READ級別
SQLSERVER: 默認(rèn)為READ_COMMITTED
臟讀?: 一個事務(wù)讀取到另一事務(wù)未提交的更新數(shù)據(jù)
不可重復(fù)讀?: 在同一事務(wù)中, 多次讀取同一數(shù)據(jù)返回的結(jié)果有所不同, 換句話說,?
后續(xù)讀取可以讀到另一事務(wù)已提交的更新數(shù)據(jù). 相反, "可重復(fù)讀"在同一事務(wù)中多次
讀取數(shù)據(jù)時, 能夠保證所讀數(shù)據(jù)一樣, 也就是后續(xù)讀取不能讀到另一事務(wù)已提交的更新數(shù)據(jù)
幻讀?: 一個事務(wù)讀到另一個事務(wù)已提交的insert數(shù)據(jù)
注意的幾點(diǎn):
1 @Transactional 只能被應(yīng)用到public方法上, 對于其它非public的方法,如果標(biāo)記了@Transactional也不會報錯,但方法沒有事務(wù)功能.
2用 spring 事務(wù)管理器,由spring來負(fù)責(zé)數(shù)據(jù)庫的打開,提交,回滾.默認(rèn)遇到運(yùn)行期例外(throw new RuntimeException("注釋");)會回滾,即遇到不受檢查(unchecked)的例外時回滾;而遇到需要捕獲的例外(throw new Exception("注釋");)不會回滾,即遇到受檢查的例外(就是非運(yùn)行時拋出的異常,編譯器會檢查到的異常叫受檢查例外或說受檢查異常)時,需我們指定方式來讓事務(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)//指定不回滾,遇到運(yùn)行期例外(throw new RuntimeException("注釋");)會回滾
public ItimDaoImpl getItemDaoImpl() {
throw new RuntimeException("注釋");
}
3、@Transactional 注解應(yīng)該只被應(yīng)用到 public 可見度的方法上。 如果你在 protected、private 或者 package-visible 的方法上使用 @Transactional 注解,它也不會報錯, 但是這個被注解的方法將不會展示已配置的事務(wù)設(shè)置。
4、@Transactional 注解可以被應(yīng)用于接口定義和接口方法、類定義和類的 public 方法上。然而,請注意僅僅 @Transactional 注解的出現(xiàn)不足于開啟事務(wù)行為,它僅僅 是一種元數(shù)據(jù),能夠被可以識別 @Transactional 注解和上述的配置適當(dāng)?shù)木哂惺聞?wù)行為的beans所使用。上面的例子中,其實(shí)正是 <tx:annotation-driven/>元素的出現(xiàn) 開啟 了事務(wù)行為。
5、Spring團(tuán)隊(duì)的建議是你在具體的類(或類的方法)上使用 @Transactional 注解,而不要使用在類所要實(shí)現(xiàn)的任何接口上。你當(dāng)然可以在接口上使用 @Transactional 注解,但是這將只能當(dāng)你設(shè)置了基于接口的代理時它才生效。因?yàn)樽⒔馐?不能繼承 的,這就意味著如果你正在使用基于類的代理時,那么事務(wù)的設(shè)置將不能被基于類的代理所識別,而且對象也將不會被事務(wù)代理所包裝(將被確認(rèn)為嚴(yán)重的)。因 此,請接受Spring團(tuán)隊(duì)的建議并且在具體的類上使用 @Transactional 注解。
轉(zhuǎn)載于:https://blog.51cto.com/fengcl/1706397
創(chuàng)作挑戰(zhàn)賽新人創(chuàng)作獎勵來咯,堅(jiān)持創(chuàng)作打卡瓜分現(xiàn)金大獎總結(jié)
以上是生活随笔為你收集整理的Spring事务处理,以及Spring事务的传播属性和隔离级别的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: IIS功能添加
- 下一篇: HDU 1789 Doing Homew