java事務異常——Spring事務異常回滾,捕獲異常不拋出就不會回滾


參考:https://blog.csdn.net/lee_sire/article/details/72904822

 

 

異常的分類:
java.lang.Throwable
1.Error錯誤:JVM內部的嚴重問題。無法恢復。程序人員不用處理。
2.Exception異常:普通的問題。通過合理的處理,程序還可以回到正常執行流程。要求編程人員要進行處理。
3.RuntimeException:也叫非受檢異常(unchecked exception).這類異常是編程人員的邏輯問題。應該承擔責任。Java編譯器不進行強制要求處理。 也就是說,這類異常再程序中,可以進行處理,也可以不處理。
4.受檢異常(checked exception).這類異常是由一些外部的偶然因素所引起的。Java編譯器強制要求處理。也就是說,程序必須進行對這類異常進行處理。

常見異常:
1)非受檢的:NullPointerException,ClassCastException,ArrayIndexsOutOfBoundsException,ArithmeticException(算術異常,除0溢出)
2)受檢:Exception,FileNotFoundException,IOException,SQLException.

最近遇到了事務不回滾的情況,我還考慮說JPA的事務有bug? 我想多了.......  
  為了打印清楚日志,很多方法我都加tyr catch,在catch中打印日志。但是這邊情況來了,當這個方法異常時候 日志是打印了,但是加的事務卻沒有回滾。

  例:  
   類似這樣的方法不會回滾 (一個方法出錯,另一個方法不會回滾) :  

[html]  view plain  copy
 
 
 
  1. if(userSave){          
  2.     try {         
  3.         userDao.save(user);          
  4.         userCapabilityQuotaDao.save(capabilityQuota);         
  5.      } catch (Exception e) {          
  6.         logger.info("能力開通接口,開戶異常,異常信息:"+e);         
  7.      }         
  8.  }  


下面的方法回滾(一個方法出錯,另一個方法會回滾):

 

 

[html]  view plain  copy
 
 
 
  1. if(userSave){         
  2.      try {          
  3.         userDao.save(user);          
  4.         userCapabilityQuotaDao.save(capabilityQuota);         
  5.        } catch (Exception e) {         
  6.         logger.info("能力開通接口,開戶異常,異常信息:"+e);          
  7.         throw new RuntimeException();         
  8.      }          
  9. }  

 

 
 
  1. if(userSave){          
  2.     try {          
  3.         userDao.save(user);          
  4.         userCapabilityQuotaDao.save(capabilityQuota);          
  5.     } catch (Exception e) {          
  6.         logger.info("能力開通接口,開戶異常,異常信息:"+e);          
  7.         TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();         
  8.     }         
  9.  }  

 

為什么不會滾呢??是對spring的事務機制不明白。!! 
   *****默認spring事務只在發生未被捕獲的 runtimeexcetpion時才回滾。 ****** 
   spring aop  異常捕獲原理:被攔截的方法需顯式拋出異常,並不能經任何處理,這樣aop代理才能捕獲到方法的異常,才能進行回滾,默認情況下aop只捕獲runtimeexception的異常,但可以通過  。
    
配置來捕獲特定的異常並回滾  
  換句話說在service的方法中不使用try catch 或者在catch中最后加上throw new runtimeexcetpion(),這樣程序異常時才能被aop捕獲進而回滾
  解決方案: 
  方案1.例如service層處理事務,那么service中的方法中不做異常捕獲,或者在catch語句中最后增加throw new RuntimeException()語句,以便讓aop捕獲異常再去回滾,並且在service上層(webservice客戶端,view層action)要繼續捕獲這個異常並處理
  方案2.在service層方法的catch語句中增加:TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();語句,手動回滾,這樣上層就無需去處理異常(現在項目的做法,方法上需要加

@Transactional


免責聲明!

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



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