關(guān)于SpringBoot的異常回滾和事務(wù)的使用詳解
Springboot中事務(wù)的使用:
1、啟動(dòng)類加上@EnableTransactionManagement注解,開啟事務(wù)支持(其實(shí)默認(rèn)是開啟的)。
2、在使用事務(wù)的public(只有public支持事務(wù))方法(或者類-相當(dāng)于該類的所有public方法都使用)加上@Transactional注解。
在實(shí)際使用中一般是在service中使用@Transactional,那么對(duì)于controller->service流程中:
如果controller未開啟事務(wù),service中開始了事務(wù),service成功執(zhí)行,controller在之后的運(yùn)行中出現(xiàn)異常(錯(cuò)誤),不會(huì)自動(dòng)回滾。
也就是說,只有在開啟事務(wù)的方法中出現(xiàn)異常(默認(rèn)只有非檢測(cè)性異常才生效-RuntimeException )(錯(cuò)誤-Error)才會(huì)自動(dòng)回滾。
如果想要對(duì)拋出的任何異常都進(jìn)行自動(dòng)回滾(而不是只針對(duì)RuntimeException),只需要在使用@Transactional(rollbackFor = Exception.class)即可。
開啟事務(wù)的方法中事務(wù)回滾的情況:
①未發(fā)現(xiàn)的異常,程序運(yùn)行過程中自動(dòng)拋出RuntimeException或者其子類,程序終止,自動(dòng)回滾。
②使用TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();進(jìn)行手動(dòng)回滾。
③注意:如果在try-catch語句中對(duì)可能出現(xiàn)的異常(RuntimeException)進(jìn)行了處理,沒有再手動(dòng)throw異常,spring認(rèn)為該方法成功執(zhí)行,不會(huì)進(jìn)行回滾,此時(shí)需要調(diào)用②中方法進(jìn)行手動(dòng)回滾 (java 框架項(xiàng)目案例:www.fhadmin.org)
另外,如果try-catch語句在finally中進(jìn)行了return操作,那么catch中手動(dòng)拋出的異常也會(huì)被覆蓋,同樣不會(huì)自動(dòng)回滾。
//不會(huì)自動(dòng)回滾 try{ throw new RuntimeException(); }catch(RuntimeException e){ e.printStackTrace(); }finally{ } //會(huì)自動(dòng)回滾 try{ throw new RuntimeException(); }catch(RuntimeException e){ e.printStackTrace(); throw new RuntimeException(); }finally{ }
Springboot @Transactional 事務(wù)不回滾
一、異常捕獲的原因
- 這里Exception異常,他又分為運(yùn)行時(shí)異常RuntimeException和非運(yùn)行時(shí)異常
- 可查的異常(checked exceptions):Exception下除了RuntimeException外的異常
- 不可查的異常(unchecked exceptions):RuntimeException及其子類和錯(cuò)誤(Error)
- 異常checked例外也回滾:在整個(gè)方法前加上 @Transactional(rollbackFor=Exception.class)
- 異常unchecked例外不回滾: @Transactional(notRollbackFor=RunTimeException.class)
- 如果異常被try{}catch{}了,事務(wù)就不回滾了,如果想讓事務(wù)回滾必須再往外拋try{}catch{throw Exception}
二、數(shù)據(jù)庫引擎不支持回滾(使用MYSQL就很可能是這個(gè)原因)
- Mysql數(shù)據(jù)庫有兩種引擎,注意要使用支持事務(wù)的引擎,比如innodb,如果是MyISAM,事務(wù)是不起作用的。
- 使用springboot的jpa自動(dòng)創(chuàng)建庫表的時(shí)候,默認(rèn)使用MyISAM引擎,可以檢查庫表查看引擎。
- 修改配置:
spring: jpa: hibernate: ddl-auto: update naming: physical-strategy: org.hibernate.boot.model.naming.PhysicalNamingStrategyStandardImpl #按字段名字建表 show-sql: true database: mysql database-platform: org.hibernate.dialect.MySQL5InnoDBDialect #使用innodb引擎建表
三、發(fā)生了自調(diào)用情況
spring的數(shù)據(jù)庫事務(wù)調(diào)用的實(shí)現(xiàn)原理是AOP,而AOP的原理是動(dòng)態(tài)代理,在自調(diào)用的過程中,是類自身的調(diào)用,而不是代理對(duì)象去調(diào)用,那么不會(huì)產(chǎn)生AOP,沒有AOP,意味著@Transactional不會(huì)被切面捕獲,這樣spring就不能把你的代碼植入到約定的流程中,于是就產(chǎn)生了事務(wù)回滾失敗。
解決方案:
1、將涉及到事務(wù)的處理都放入一個(gè)方法中,由其他類調(diào)用。
2、如果非要在本類中調(diào)用,那么需要在本類中生成本類的bean對(duì)象,由這個(gè)bean對(duì)象來調(diào)用,實(shí)現(xiàn)方式也有多種:
- 自注入
- 通過SpringContextHolder獲取bean,即注入ApplicationContext,從ApplicationContext獲取bean
- 通過AopContext獲取當(dāng)前類的代理對(duì)象,注意此種方式需要開啟expose-proxy=“true”,可通過注解開啟@EnableAspectJAutoProxy(exposeProxy =true)
最后一種方式的偽代碼:
@Service public class OrderService { public void insert() { OrderService proxy = (OrderService) AopContext.currentProxy(); proxy.insertOrder(); } @Transactional public void insertOrder() { //SQL操作 } }
因?yàn)锳opContext默認(rèn)是不暴露當(dāng)前代理類的,所以要@EnableAspectJAutoProxy(exposeProxy =true)或者<aop:aspectj-autoproxy expose-proxy=“true”/>:
四、補(bǔ)充:
在Spring 的AOP實(shí)現(xiàn)有兩種代理方式:
- Java動(dòng)態(tài)代理 :通過反射生成一個(gè)實(shí)現(xiàn)了代理方法的匿名類來完成代理
- cglib代理 :通過Asm修改字節(jié)碼文件,生成一個(gè)子類來完成代理
Spring在項(xiàng)目中會(huì)根據(jù)被代理對(duì)象是否實(shí)現(xiàn)了接口來自動(dòng)切換上述兩種代理方式
所以private方法也不能實(shí)現(xiàn)事務(wù)
到此這篇關(guān)于關(guān)于SpringBoot的異?;貪L和事務(wù)的使用詳解的文章就介紹到這了,更多相關(guān)SpringBoot異?;貪L和事務(wù)的使用內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Spring使用注解進(jìn)行引用類型的自動(dòng)裝配逐步分析
自動(dòng)裝配是springboot的核心,一般提到自動(dòng)裝配就會(huì)和springboot聯(lián)系在一起。實(shí)際上Spring Framework早就實(shí)現(xiàn)了這個(gè)功能。Spring Boot只是在其基礎(chǔ)上,通過SPI的方式,做了進(jìn)一步優(yōu)化2023-03-03springboot+vue實(shí)現(xiàn)七牛云頭像的上傳
本文將介紹如何在Spring Boot項(xiàng)目中利用七牛云進(jìn)行圖片上傳并將圖片存儲(chǔ)在云存儲(chǔ)中,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2023-08-08SpringBoot獲取Request對(duì)象的常見方法
HttpServletRequest 簡(jiǎn)稱 Request,它是一個(gè) Servlet API 提供的對(duì)象,用于獲取客戶端發(fā)起的 HTTP 請(qǐng)求信息,那么在SpringBoot中,獲取 Request對(duì)象的方法有哪些呢,本文小編將給大家講講SpringBoot獲取Request對(duì)象的常見方法2023-08-08spring boot集成rabbitmq的實(shí)例教程
這篇文章主要給大家介紹了關(guān)于spring boot集成rabbitmq的相關(guān)資料,springboot集成RabbitMQ非常簡(jiǎn)單,文中通過示例代碼介紹的非常詳細(xì),需要的朋友們可以參考借鑒,下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧。2017-11-11JavaEE中struts2實(shí)現(xiàn)文件上傳下載功能實(shí)例解析
這篇文章主要為大家詳細(xì)介紹了JavaEE中struts2實(shí)現(xiàn)文件上傳下載功能實(shí)例,感興趣的小伙伴們可以參考一下2016-05-05簡(jiǎn)單了解java標(biāo)識(shí)符的作用和命名規(guī)則
這篇文章主要介紹了簡(jiǎn)單了解java標(biāo)識(shí)符的作用和命名規(guī)則,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-01-01