首頁 > 軟體

解決try-catch捕獲異常資訊後Spring事務失效的問題

2021-06-21 16:00:23

一、首先在Spring Boot專案中,手動新增異常方法進行測試

@Transactional(rollbackFor=Exception.class) //表示此方法有異常時觸發Spring事務
@Override
public CommonResult<User> saveUser(User user) {
    int insert = baseMapper.insert(user);
    try {
        // 新增異常,並進行捕獲
        int a = 10/0;
    }catch (Exception e){
        logger.info("列印異常資訊:"+e);
        return CommonResult.commentFailure("伺服器異常,事務回滾");
    }
    if(insert > 0){
        return CommonResult.commentSuccess(user);
    }else {
        return CommonResult.commentFailure("新增失敗");
    }
}

1、一個新增資訊的實現類方法上,此處我們加了Spring的事務。

2、問題:一個方法報異常(int a = 10/0)進行了異常捕獲,另一個方法不會回滾(insert新增方法)

這是什麼情況呢,相當於Spring事務策略失效了。

try-catch捕獲了異常後,這種業務方法也就等於脫離了spring事務的管理,因為沒有任何異常會從業務方法中丟擲,全被捕獲並「吞掉」,導致spring異常丟擲觸發事務回滾策略失效。

通俗的來說:預設spring事務只在發生未被捕獲的 runtimeexcetpion或error時才回滾。

二、處理方案一

spring aop 異常捕獲進而回滾。在catch中最後加上throw new runtimeexcetpion(),這樣程式異常時才能被aop捕獲進而回滾,缺點是無法return異常資訊提示,前端使用者互動效果不佳

@Transactional(rollbackFor=Exception.class)  //表示此方法有異常時觸發Spring事務
@Override
public CommonResult<User> saveUser(User user) {
    int insert = baseMapper.insert(user);
    try {
        // 新增異常,並進行捕獲
        int a = 10/0;
    }catch (Exception e){
        logger.info("異常資訊:"+e);
        // 方案一:spring aop 異常捕獲
        throw new RuntimeException();
    }
    if(insert > 0){
        return CommonResult.commentSuccess(user);
    }else {
        return CommonResult.commentFailure("新增失敗");
    }
}

三、處理方案二

就是讓一個方法報異常,另一個方法回滾,這樣才能真正的觸發Spring事務回滾策略。

catch語句中增加:

TransactionAspectSupport.currentTransactionStatus().setRollbackOnly(); //手動回滾,這樣上層就無需去處理異常了

完整程式碼:

@Transactional(rollbackFor=Exception.class) //表示此方法有異常時觸發Spring事務
@Override
 public CommonResult<User> saveUser(User user) {
     int insert = baseMapper.insert(user);
     try {
         // 新增異常,並進行捕獲
         int a = 10/0;
     }catch (Exception e){
         logger.info("異常資訊:"+e);
         // 方案二:手動回滾
         TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();
         return CommonResult.commentFailure("伺服器異常,事務回滾");
     }
     if(insert > 0){
         return CommonResult.commentSuccess(user);
     }else {
         return CommonResult.commentFailure("新增失敗");
     }
 }

四、如過需要手動進行手動回滾的業務方法比較多,我們可以寫一個公共的工具類

SpringRollBackUtil.java

public class SpringRollBackUtil {
    /**
     * 事務回滾機制
     */
    public static void rollBack() {
        try {
            TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

只需呼叫方法即可

// 方案三:公共工具類 手動回滾
SpringRollBackUtil.rollBack();

Spring mvc:事務引起的try/catch失效

在測試一個介面時,發現一個奇怪的現象:

該介面使用@ResponseBody註解返回json格式資料,並且使用try/catch包括全部邏輯程式碼,debug後發現返回資料沒有任何錯誤,只包含一段因產生異常導致的錯誤提示字串,但是chrome瀏覽器network卻顯示http狀態碼為500。

最後發現在該RequestMapping方法上還有一個註解@Transactional,去除ok。

以上為個人經驗,希望能給大家一個參考,也希望大家多多支援it145.com。


IT145.com E-mail:sddin#qq.com