解決SpringBoot中使用@Transactional注解遇到的問題
使用@Transactional注解遇到的問題
1、不建議在接口上添加@Transactional注解
一般在service類標(biāo)簽上添加@Transactional即可
2、@Transactional注解
只能應(yīng)用到public可見度的方法上。如果應(yīng)用到protected、private或者package可見度的方法上時(shí),不會(huì)報(bào)錯(cuò),但事務(wù)也不會(huì)起作用
3、默認(rèn)情況下
spring會(huì)對(duì)uncheck異常進(jìn)行事務(wù)回滾的;如果是checked異常則不會(huì)回滾,可添加注解 @Transactional(rollbackFor=Exception.class) 是的checked異?;貪L。
- uncheck異常:java里面將派生于Error或者RuntimeException(比如空指針,1/0)的異常
- checked異常:其他繼承自java.lang.Exception得異常統(tǒng)稱為Checked Exception,如IOException、TimeoutException等
4、數(shù)據(jù)庫引擎需要支持事務(wù)管理
如果是mysql,注意表要使用事務(wù)的引擎,比如innodb,如果是myisam,事務(wù)不會(huì)起作用的
5、同一類中methodA()方法
沒有@Transactional 注解,在其內(nèi)部調(diào)用有@Transactional 注解的方法,有@Transactional 注解的方法methodB()的事務(wù)被忽略,不會(huì)開啟新的事務(wù),也不會(huì)發(fā)生回滾。例如:
@Service
public class TransactionService {
public void methodA(){
this.methodB();
}
@Transactional
public void methodB(){
}
}
原因:Spring采用動(dòng)態(tài)代理(AOP)實(shí)現(xiàn)對(duì)bean的管理和切片,它為我們的每個(gè)class生成一個(gè)代理對(duì)象。只有在代理對(duì)象之間進(jìn)行調(diào)用時(shí),可以觸發(fā)切面邏輯。在 Spring 的 AOP 代理下,只有目標(biāo)方法由外部調(diào)用,目標(biāo)方法才由 Spring 生成的代理對(duì)象來管理。
詳解:Spring的事務(wù)管理是通過AOP實(shí)現(xiàn)的,其AOP的實(shí)現(xiàn)對(duì)于非final類是通過cglib這種方式,即生成當(dāng)前類的一個(gè)子類作為代理類,然后在調(diào)用其下的方法時(shí),會(huì)判斷這個(gè)方法有沒有@Transactional注解,如果有的話,則會(huì)開啟一個(gè)新的事務(wù),并通過動(dòng)態(tài)代理實(shí)現(xiàn)事務(wù)管理(攔截方法調(diào)用,執(zhí)行事務(wù)等切面)。
當(dāng)methodA()中調(diào)用methodB()時(shí),并不是使用的代理對(duì)象,而是普通的javabean,從而導(dǎo)致this.methodB()時(shí)也不是代碼對(duì)象,從而導(dǎo)致@Transactional失敗,即發(fā)現(xiàn)methodA()上并沒有@Transactional注解,所以整個(gè)AOP代理過程(事務(wù)管理)不會(huì)發(fā)生。
解決辦法:
1、把這兩個(gè)方法分開到不同的類中;
2、把注解@Transactional加到類名上面去;
3、把注解@Transactional加到methodA()方法上,methodB()不添加注解,在調(diào)用methodB()時(shí)兩個(gè)方法的事務(wù)都會(huì)生效,因?yàn)閙ethodA()默認(rèn)的事務(wù)傳播屬性為PROPAGATION_REQUIRED,此時(shí)methodB()會(huì)加入到methodA()中
@Service
public class TransactionService {
@Transactional
public void methodA(){
this.methodB();
}
public void methodB(){
}
}
4、獲取本對(duì)象的代理對(duì)象,再進(jìn)行調(diào)用。具體操作如:
- Spring-content.xml上下文中,增加配置:<aop:aspectj-autoproxy expose-proxy="true"/>
- 在TransactionService 中,用(transactionService )(AopContext.currentProxy()),獲取到TransactionService 的代理類,再調(diào)用事務(wù)方法,強(qiáng)行經(jīng)過代理類,激活事務(wù)切面。
springboot 注解transactional失效
1、在方法中捕獲了異常
那么該方法事務(wù)不會(huì)回滾。本類中A方法(有事務(wù)注解)調(diào)用其他類中B方法(有事務(wù)注解,該類在容器中),B方法中捕獲了異常,事務(wù)失效不可以回滾,在B方法中的catch里進(jìn)行異常拋出則可以正?;貪L。
本類中A方法(有事務(wù)注解)調(diào)用其他類中B方法(有事務(wù)注解,該類在容器中),A方法中捕獲了異常,spring事務(wù)機(jī)制檢測到異常,進(jìn)行回滾。
所以最好在業(yè)務(wù)層catch中拋出異常讓控制層捕獲。
2、spring中事務(wù)是代理模式
在該類中A方法調(diào)用本類中B方法,A方法上沒有事務(wù)注解,B方法上有事務(wù)注解,則B方法中出現(xiàn)異常,B方法事務(wù)不會(huì)回滾,A方法沒有事務(wù)注解也不會(huì)回滾。
3、A方法如果有事務(wù)注解
調(diào)用了本類B方法,B方法出現(xiàn)異常也整體可回滾。
4、本類中A方法調(diào)用
其他類中B方法(有事務(wù)注解,該類在容器中),B方法發(fā)生異常,可以回滾。
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
Java?webservice的POST和GET請(qǐng)求調(diào)用方式
這篇文章主要介紹了Java?webservice的POST和GET請(qǐng)求調(diào)用方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-03-03
MyBatis3傳遞多個(gè)參數(shù)(Multiple Parameters)
這篇文章主要介紹了MyBatis3傳遞多個(gè)參數(shù),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-07-07
Mybatis-Plus最優(yōu)化持久層開發(fā)過程
Mybatis-plus(簡稱MP)是一個(gè)Mybatis的增強(qiáng)工具,在mybatis的基礎(chǔ)上只做增強(qiáng)不做改變,提高效率,自動(dòng)生成單表的CRUD功能,這篇文章主要介紹了Mybatis-Plus最優(yōu)化持久層開發(fā),需要的朋友可以參考下2024-07-07
SpringBoot實(shí)現(xiàn)海量數(shù)據(jù)高效實(shí)時(shí)搜索功能
我們都知道隨著業(yè)務(wù)系統(tǒng)的發(fā)展和使用,數(shù)據(jù)庫存儲(chǔ)的業(yè)務(wù)數(shù)據(jù)量會(huì)越來越大,逐漸成為了業(yè)務(wù)系統(tǒng)的瓶頸,本文給大家介紹了Spring Boot業(yè)務(wù)系統(tǒng)如何實(shí)現(xiàn)海量數(shù)據(jù)高效實(shí)時(shí)搜索,文中有詳細(xì)的代碼示例,需要的朋友可以參考下2023-10-10
解決Error:(1,?1)?java:?非法字符:?'\ufeff'問題
這篇文章主要介紹了解決Error:(1,?1)?java:?非法字符:?'\ufeff'問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-11-11
Java跳出當(dāng)前的多重嵌套循環(huán)的五種方法
在Java編程中,跳出多重嵌套循環(huán)可以使用break語句、標(biāo)號(hào)與break組合、return語句、標(biāo)志變量和異常處理五種方法,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2024-10-10
Java日常練習(xí)題,每天進(jìn)步一點(diǎn)點(diǎn)(22)
下面小編就為大家?guī)硪黄狫ava基礎(chǔ)的幾道練習(xí)題(分享)。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧,希望可以幫到你2021-07-07

