聊聊@Transactional事務(wù)注意點(diǎn),你學(xué)會(huì)了嗎?
這里面有幾點(diǎn)需要大家留意:
A. 一個(gè)功能是否要事務(wù),必須納入設(shè)計(jì)、編碼考慮。不能僅僅完成了基本功能就ok。
B. 如果加了事務(wù),必須做好開發(fā)環(huán)境測(cè)試(測(cè)試環(huán)境也盡量觸發(fā)異常、測(cè)試回滾),確保事務(wù)生效。
C. 以下列了事務(wù)使用過(guò)程的注意事項(xiàng),請(qǐng)大家留意。
1.不要在接口上聲明@Transactional ,而要在具體類的方法上使用 @Transactional 注解,否則注解可能無(wú)效。
2.不要圖省事,將@Transactional放置在類級(jí)的聲明中,放在類聲明,會(huì)使得所有方法都有事務(wù)。故@Transactional應(yīng)該放在方法級(jí)別,不需要使用事務(wù)的方法,就不要放置事務(wù),比如查詢方法。否則對(duì)性能是有影響的。
3.使用了@Transactional的方法,對(duì)同一個(gè)類里面的方法調(diào)用, @Transactional無(wú)效。比如有一個(gè)類Test,它的一個(gè)方法A,A再調(diào)用Test本類的方法B(不管B是否public還是private),但A沒(méi)有聲明注解事務(wù),而B有。則外部調(diào)用A之后,B的事務(wù)是不會(huì)起作用的。(經(jīng)常在這里出錯(cuò))
4.使用了@Transactional的方法,只能是public,@Transactional注解的方法都是被外部其他類調(diào)用才有效,故只能是public。道理和上面的有關(guān)聯(lián)。故在 protected、private 或者 package-visible 的方法上使用 @Transactional 注解,它也不會(huì)報(bào)錯(cuò),但事務(wù)無(wú)效。
5.spring的事務(wù)在拋異常的時(shí)候會(huì)回滾,如果是catch捕獲了,事務(wù)無(wú)效。可以在catch里面加上throw new RuntimeException();
6.最后有個(gè)關(guān)鍵的一點(diǎn):
和鎖同時(shí)使用需要注意
由于Spring事務(wù)是通過(guò)AOP實(shí)現(xiàn)的,所以在方法執(zhí)行之前會(huì)有開啟事務(wù),之后會(huì)有提交事務(wù)邏輯。
而synchronized代碼塊執(zhí)行是在事務(wù)之內(nèi)執(zhí)行的,可以推斷在synchronized代碼塊執(zhí)行完時(shí),事務(wù)還未提交,其他線程進(jìn)入synchronized代碼塊后,讀取的數(shù)據(jù)不是最新的。
所以必須使synchronized鎖的范圍大于事務(wù)控制的范圍,
把synchronized加到Controller層或者大于事務(wù)邊界的調(diào)用層!