Spring @Transactional注解不回滾不起作用無效


一、特性
先來了解一下@Transactional注解的特性吧,可以更好排查問題

  1. service類標簽(一般不建議在接口上)上添加@Transactional,可以將整個類納入spring事務管理,在每個業務方法執行時都會開啟一個事務,不過這些事務采用相同的管理方式。

  2. @Transactional 注解只能應用到 public 可見度的方法上。 如果應用在protected、private或者 package可見度的方法上,也不會報錯,不過事務設置不會起作用。

  3. 默認情況下,spring會對unchecked異常進行事務回滾;如果是checked異常則不回滾。

辣么什么是checked異常,什么是unchecked異常?

  • java里面將派生於Error或者RuntimeException(比如空指針,1/0)的異常稱為unchecked異常,
  • 其他繼承自java.lang.Exception得異常統稱為Checked Exception,如IOException、TimeoutException等
  • 再通俗一點:你寫代碼出現的空指針等異常,會被回滾,文件讀寫,網絡出問題,spring就沒法回滾了。

4。 只讀事務:
如果事務在try{}catch(Exception e){e.printStackTrace();}中跑,並且catch中只是打印e的話,那么事務不會rollback。因為異常被catch掉了,框架不知道發生了異常。
如果想要rollback,可以加上@Transactional(propagation=Propagation.NOT_SUPPORTED,readOnly=true)

①在方法上添加 throws Exception,將方法中出現的異常拋出給spring事務,
②去掉方法體中的try catch
③catch (Exception e) { throw e;}繼續向上拋,目的是讓spring事務捕獲這個異常。

只讀標志只在事務啟動時應用,否則即使配置也會被忽略。
啟動事務會增加線程開銷,數據庫因共享讀取而鎖定(具體跟數據庫類型和事務隔離級別有關)。通常情況下,僅是讀取數據時,不必設置只讀事務而增加額外的系統開銷。

Spring管理事務默認回滾的異常是什么?
答案是 RuntimeException或者Error,即unchecked Exception 詳http://blog.csdn.net/u012557814/article/details/50685374

只有spring事務捕獲到Exception異常后,@Transactional(rollbackFor=Exception.class),才會起到應有的作用;catch (Exception e) { e.printStackTrace(); }這句是捕獲try中出現的Exception然后將異常信息打印出來,僅僅是打印出來,然后什么也沒干。

  • @Transactional(timeout = 60,rollbackFor=Exception.class)與@Transactional(timeout = 60)區別: rollbackFor=Exception.class的作用是讓checked例外也回滾,即讓編譯錯誤也回滾。

需要注明的是:
1 讓checked例外也回滾:在整個方法前加上 @Transactional(rollbackFor=Exception.class)
2 讓unchecked例外不回滾: @Transactional(notRollbackFor=RunTimeException.class)
checked exception是編譯時的錯誤,Unchecked exception是運行時錯誤,詳見http://meng702.iteye.com/blog/998379

二、事務傳播模式

Propagation枚舉了多種事務傳播模式,部分列舉如下:

  1. REQUIRED(默認模式):業務方法需要在一個容器里運行。如果方法運行時,已經處在一個事務中,那么加入到這個事務,否則自己新建一個新的事務。

  2. NOT_SUPPORTED:聲明方法不需要事務。如果方法沒有關聯到一個事務,容器不會為他開啟事務,如果方法在一個事務中被調用,該事務會被掛起,調用結束后,原先的事務會恢復執行。

  3. REQUIRESNEW:不管是否存在事務,該方法總匯為自己發起一個新的事務。如果方法已經運行在一個事務中,則原有事務掛起,新的事務被創建。

  4. MANDATORY:該方法只能在一個已經存在的事務中執行,業務方法不能發起自己的事務。如果在沒有事務的環境下被調用,容器拋出例外。

  5. SUPPORTS:該方法在某個事務范圍內被調用,則方法成為該事務的一部分。如果方法在該事務范圍外被調用,該方法就在沒有事務的環境下執行。

  6. NEVER:該方法絕對不能在事務范圍內執行。如果在就拋例外。只有該方法沒有關聯到任何事務,才正常執行。NESTED:如果一個活動的事務存在,則運行在一個嵌套的事務中。如果沒有活動事務,則按REQUIRED屬性執行。它使用了一個單獨的事務,這個事務擁有多個可以回滾的保存點。內部事務的回滾不會對外部事務造成影響。它只對DataSourceTransactionManager事務管理器起效。

 

三、解決Transactional注解不回滾

  1. 檢查你方法是不是public的。

  2. 你的異常類型是不是unchecked異常。

    如果我想check異常也想回滾怎么辦,注解上面寫明異常類型即可。
    @Transactional(rollbackFor=Exception.class)
    類似的還有norollbackFor,自定義不回滾的異常。

  3. 數據庫引擎要支持事務,如果是mysql,注意表要使用支持事務的引擎,比如innodb,如果是myisam,事務是不起作用的。

  4. 是否開啟了對注解的解析

    <tx:annotation-driven transaction-manager="transactionManager" proxy-target-class="true"/>

  5. spring是否掃描到你這個包,如下是掃描到org.test下面的包

 

 

 

 

轉載:https://blog.csdn.net/qq_34495753/article/details/79150996


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM