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

一、首先在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。

以上為個人經驗,希望能給大傢一個參考,也希望大傢多多支持WalkonNet。

推薦閱讀: