javascript
SpringBoot 异常回滚 事务的使用___Springboot @Transactional 事务不回滚
Springboot中事務(wù)的使用:
1、啟動類加上@EnableTransactionManagement注解,開啟事務(wù)支持(其實默認(rèn)是開啟的)。
2、在使用事務(wù)的public(只有public支持事務(wù))方法(或者類-相當(dāng)于該類的所有public方法都使用)加上@Transactional注解。
在實際使用中一般是在service中使用@Transactional,那么對于controller->service流程中:
如果controller未開啟事務(wù),service中開始了事務(wù),service成功執(zhí)行,controller在之后的運行中出現(xiàn)異常(錯誤),不會自動回滾。
也就是說,只有在開啟事務(wù)的方法中出現(xiàn)異常(默認(rèn)只有非檢測性異常才生效-RuntimeException )(錯誤-Error)才會自動回滾。
如果想要對拋出的任何異常都進行自動回滾(而不是只針對RuntimeException),只需要在使用@Transactional(rollbackFor = Exception.class)即可。
開啟事務(wù)的方法中事務(wù)回滾的情況:
①未發(fā)現(xiàn)的異常,程序運行過程中自動拋出RuntimeException或者其子類,程序終止,自動回滾。
②使用TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();進行手動回滾。
③注意:如果在try-catch語句中對可能出現(xiàn)的異常(RuntimeException)進行了處理,沒有再手動throw異常,spring認(rèn)為該方法成功執(zhí)行,不會進行回滾,此時需要調(diào)用②中方法進行手動回滾 (java 框架項目案例:www.fhadmin.org)
另外,如果try-catch語句在finally中進行了return操作,那么catch中手動拋出的異常也會被覆蓋,同樣不會自動回滾。
//不會自動回滾 try{throw new RuntimeException(); }catch(RuntimeException e){e.printStackTrace(); }finally{} //會自動回滾 try{throw new RuntimeException(); }catch(RuntimeException e){e.printStackTrace();throw new RuntimeException(); }finally{}Springboot @Transactional 事務(wù)不回滾
一、異常捕獲的原因
二、數(shù)據(jù)庫引擎不支持回滾(使用MYSQL就很可能是這個原因)
Mysql數(shù)據(jù)庫有兩種引擎,注意要使用支持事務(wù)的引擎,比如innodb,如果是MyISAM,事務(wù)是不起作用的。
使用springboot的jpa自動創(chuàng)建庫表的時候,默認(rèn)使用MyISAM引擎,可以檢查庫表查看引擎。
修改配置:
spring: jpa:hibernate:ddl-auto: updatenaming:physical-strategy: org.hibernate.boot.model.naming.PhysicalNamingStrategyStandardImpl #按字段名字建表show-sql: truedatabase: mysqldatabase-platform: org.hibernate.dialect.MySQL5InnoDBDialect #使用innodb引擎建表三、發(fā)生了自調(diào)用情況
spring的數(shù)據(jù)庫事務(wù)調(diào)用的實現(xiàn)原理是AOP,而AOP的原理是動態(tài)代理,在自調(diào)用的過程中,是類自身的調(diào)用,而不是代理對象去調(diào)用,那么不會產(chǎn)生AOP,沒有AOP,意味著@Transactional不會被切面捕獲,這樣spring就不能把你的代碼植入到約定的流程中,于是就產(chǎn)生了事務(wù)回滾失敗。
解決方案:
1、將涉及到事務(wù)的處理都放入一個方法中,由其他類調(diào)用。
2、如果非要在本類中調(diào)用,那么需要在本類中生成本類的bean對象,由這個bean對象來調(diào)用,實現(xiàn)方式也有多種:
- 自注入
- 通過SpringContextHolder獲取bean,即注入ApplicationContext,從ApplicationContext獲取bean
- 通過AopContext獲取當(dāng)前類的代理對象,注意此種方式需要開啟expose-proxy=“true”,可通過注解開啟@EnableAspectJAutoProxy(exposeProxy =true)
最后一種方式的偽代碼:
@Service public class OrderService {public void insert() {OrderService proxy = (OrderService) AopContext.currentProxy();proxy.insertOrder();}@Transactionalpublic void insertOrder() {//SQL操作} }[
因為AopContext默認(rèn)是不暴露當(dāng)前代理類的,所以要@EnableAspectJAutoProxy(exposeProxy =true)或者<aop:aspectj-autoproxy expose-proxy=“true”/>:
四、補充:
在Spring 的AOP實現(xiàn)有兩種代理方式:
- Java動態(tài)代理 :通過反射生成一個實現(xiàn)了代理方法的匿名類來完成代理
- cglib代理 :通過Asm修改字節(jié)碼文件,生成一個子類來完成代理
Spring在項目中會根據(jù)被代理對象是否實現(xiàn)了接口來自動切換上述兩種代理方式
所以private方法也不能實現(xiàn)事務(wù)
總結(jié)
以上是生活随笔為你收集整理的SpringBoot 异常回滚 事务的使用___Springboot @Transactional 事务不回滚的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: springboot系列——redisT
- 下一篇: Spring深入理解之Component