SpringBoot中Service層事務控制

SpringBoot中使用事務比較簡單,在Application啓動類上添加@EnableTransactionManagement註解,然後在service層的方法上添加@Transactional註解

@Transactional屬性 

屬性 類型 描述
value String 可選的限定描述符,指定使用的事務管理器
propagation enum: Propagation 可選的事務傳播行爲設置
isolation enum: Isolation 可選的事務隔離級別設置
readOnly boolean 讀寫或只讀事務,默認讀寫
timeout int (in seconds granularity) 事務超時時間設置
rollbackFor Class對象數組,必須繼承自Throwable 導致事務回滾的異常類數組
rollbackForClassName 類名數組,必須繼承自Throwable 導致事務回滾的異常類名字數組
noRollbackFor Class對象數組,必須繼承自Throwable 不會導致事務回滾的異常類數組
noRollbackForClassName 類名數組,必須繼承自Throwable 不會導致事務回滾的異常類名字數組

用法:

        @Transactional 可以作用於接口、接口方法、類以及類方法上。當作用於類上時,該類的所有 public 方法將都具有該類型的事務屬性,同時,我們也可以在方法級別使用該標註來覆蓋類級別的定義。

        雖然 @Transactional 註解可以作用於接口、接口方法、類以及類方法上,但是 Spring 建議不要在接口或者接口方法上使用該註解,因爲這隻有在使用基於接口的代理時它纔會生效。另外, @Transactional 註解應該只被應用到 public 方法上,這是由 Spring AOP 的本質決定的。如果你在 protected、private 或者默認可見性的方法上使用 @Transactional 註解,這將被忽略,也不會拋出任何異常。默認情況下,只有來自外部的方法調用纔會被AOP代理捕獲,也就是,類內部方法調用本類內部的其他方法並不會引起事務行爲,即使被調用方法使用@Transactional註解進行修飾

註解在方法上:

@Autowired  
private MyBatisDao dao;  
  
@Transactional  
@Override  
public void insert(Test test) {  
    dao.insert(test);  
    throw new RuntimeException("test");//拋出unchecked異常,觸發事物,回滾  
}  
@Transactional(noRollbackFor=RuntimeException.class)  
@Override  
public void insert(Test test) {  
    dao.insert(test);  
    //拋出unchecked異常,觸發事物,noRollbackFor=RuntimeException.class,不回滾  
    throw new RuntimeException("test");  
}
@Transactional(propagation=Propagation.NOT_SUPPORTED)  
@Override  
public void insert(Test test) {  
    //事物傳播行爲是PROPAGATION_NOT_SUPPORTED,以非事務方式運行,不會存入數據庫  
    dao.insert(test);  
}  

 註解在類上:當作用於類上時,該類的所有 public 方法將都具有該類型的事務屬性

@Transactional  
public class MyBatisServiceImpl implements MyBatisService {  
  
    @Autowired  
    private MyBatisDao dao;  
      
      
    @Override  
    public void insert(Test test) {  
        dao.insert(test);  
        //拋出unchecked異常,觸發事物,回滾  
        throw new RuntimeException("test");  
    }
}

注意:有時候在開發時往往出現了添加註解後卻無法回滾

原因:默認Spring事務只在發生未被捕獲的RuntimeException時纔回滾。  
Spring AOP異常捕獲原理:被攔截的方法需顯式拋出異常,並不能經任何處理,這樣AOP代理才能捕獲到方法的異常,才能進行回滾,默認情況下AOP只捕獲 RuntimeException 的異常,但可以通過配置來捕獲特定的異常並回滾  
換句話說在service的方法中不使用try...catch或者在catch中最後加上throw new runtimeexcetpion(),這樣程序異常時才能被aop捕獲進而回滾。

解決方案:

  1.在service層方法的catch語句中拋出RuntimeException異常(如在catch中需要拋出自定義的異常則只需自定義異常繼承RuntimeException即可),從而觸發Spring的異常捕獲機制進行事務回滾,但這種方式需要上層處理異常。

  2.在service層方法的catch語句中增加如下語句,手動回滾,這樣上層就無需去處理異常:

TransactionAspectSupport.currentTransactionStatus().setRollbackOnly()

 

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章