spring注解-@Transactional事務(wù)幾點注意
這里面有幾點需要大家留意:
A. 一個功能是否要事務(wù),必須納入設(shè)計、編碼考慮。不能僅僅完成了基本功能就ok。
B. 如果加了事務(wù),必須做好開發(fā)環(huán)境測試(測試環(huán)境也盡量觸發(fā)異常、測試回滾),確保事務(wù)生效。
C. 以下列了事務(wù)使用過程的注意事項,請大家留意。
1.?不要在接口上聲明@Transactional ,而要在具體類的方法上使用 @Transactional 注解,否則注解可能無效。
2.不要圖省事,將@Transactional放置在類級的聲明中,放在類聲明,會使得所有方法都有事務(wù)。故@Transactional應(yīng)該放在方法級別,不需要使用事務(wù)的方法,就不要放置事務(wù),比如查詢方法。否則對性能是有影響的。
3.使用了@Transactional的方法,對同一個類里面的方法調(diào)用, @Transactional無效。比如有一個類Test,它的一個方法A,A再調(diào)用Test本類的方法B(不管B是否public還是private),但A沒有聲明注解事務(wù),而B有。則外部調(diào)用A之后,B的事務(wù)是不會起作用的。(經(jīng)常在這里出錯)
解決辦法AopContext.currentProxy().B()
4.使用了@Transactional的方法,只能是public,@Transactional注解的方法都是被外部其他類調(diào)用才有效,故只能是public。道理和上面的有關(guān)聯(lián)。故在 protected、private 或者 package-visible 的方法上使用 @Transactional 注解,它也不會報錯,但事務(wù)無效。
5.經(jīng)過在ICORE-CLAIM中測試,效果如下:
A.拋出受查異常XXXException,事務(wù)會回滾。
B.拋出運行時異常NullPointerException,事務(wù)會回滾。
C.Quartz中,execute直接調(diào)用加了@Transactional方法,可以回滾;間接調(diào)用,不會回滾。(即上文3點提到的)
D.異步任務(wù)中,execute直接調(diào)用加了@Transactional方法,可以回滾;間接調(diào)用,不會回滾。(即上文3點提到的)
E.在action中加上@Transactional,不會回滾。切記不要在action中加上事務(wù)。
F.在service中加上@Transactional,如果是action直接調(diào)該方法,會回滾,如果是間接調(diào),不會回滾。(即上文3提到的)
G.在service中的private加上@Transactional,事務(wù)不會回滾。
(總結(jié)):
在一個Service內(nèi)部:
1.非事務(wù)方法a調(diào)用事務(wù)方法b,不會開啟新的事務(wù).是因為spring采用動態(tài)代理機制來實現(xiàn)事務(wù)控制,而動態(tài)代理最終都是要調(diào)用原始對象的,而原始對象在去調(diào)用方法時,是不會再觸發(fā)代理了!要想事務(wù)有效,需用((xxxService)AopContext.currentProxy()).b()來調(diào)用,這樣b方法會回滾,但a方法里的邏輯不會回滾。
//注意:(1)需在application啟動類上加 @EnableAspectJAutoProxy(exposeProxy = true)
//注意:(2)pom文件中加org.aspectj下的aspectjrt和aspectjweaver兩個jar依賴
2.事務(wù)方法調(diào)用非事務(wù)方法,整個過程事務(wù)都生效;
3.事務(wù)方法調(diào)用事務(wù)方法,整個過程事務(wù)都生效;
兩個Service:
serviceA中的非事務(wù)方法調(diào)用另了serviceB的事務(wù)方法,serviceB中的方法邏輯會回滾,但serviceA中方法里的邏輯不會回滾,因為他沒加事務(wù)。