溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

遇到Spring雙層事務不回滾怎么辦

發布時間:2021-12-02 16:02:56 來源:億速云 閱讀:174 作者:柒染 欄目:云計算

今天就跟大家聊聊有關遇到Spring雙層事務不回滾怎么辦,可能很多人都不太了解,為了讓大家更加了解,小編給大家總結了以下內容,希望大家根據這篇文章可以有所收獲。

最近有粉絲在朋友圈問我,面試遇到了Spring雙事務不會滾問題,怎么破解。下面結合一個簡單案例,希望能解決一部分人的疑惑。

系統 A 調用系統 B 執行數據同步,系統 B 返回了錯誤提示,系統 A 需要將前邊保存的回滾掉,同時把錯誤信息向上拋。

大致代碼如下

@Service("noteService")
public class NoteServiceImpl implements NoteService {

    @Resource
    private SearchService searchService;


    @Transactional(rollbackFor = Throwable.class)
    @Override
    public CommonResponse<NoteEntity> save(NoteEntity note) {
        // 一系列 DB 操作

        try {
            searchService.sync(note);
        } catch (Exception e) {
            e.printStackTrace();
        }

        return CommonResponse.success(entity);
    }

}

@Service("searchService")
public class SearchServiceImpl implements SearchService {

    @Transactional(rollbackFor = Throwable.class)
    @Override
    public void sync(NoteEntity note) {
        // 一系列 DB 操作

        throw new RuntimeException("同步異常! [XXX]");
    }

}

@SpringBootTest
public class NoteTests {

    @Resource
    private NoteService noteService;

    @Test
    public void saveNote() {
        NoteEntity entity = new NoteEntity();
        entity.setTitle("念奴嬌赤壁懷古");
        entity.setContent("大江東去,浪淘盡,千古風流人物。故壘西邊,人道是:三國周郎赤壁。。。");
        entity.setTags("蘇軾,宋代");
        entity.setCategory("蘇軾詩詞");

        try {
            noteService.save(entity);
        } catch (Exception e) {
            e.printStackTrace();
            // FIXME 我想在這里拿到的是 同步異常! [XXX]
            // FIXME 但是這里拿到的是 Transaction silently rolled back because it has been marked as rollback-only
            System.out.println(">>>>>>>>>> " + e.getMessage());
        }
    }

}

事出有因

代碼歷史久遠,為何這樣寫已無從追溯。

納悶了一會兒,看到雙層事務,就想起了 Spring事務傳播機制,前邊理解得比較膚淺。

沒有特殊的配置,自然是走默認的事務傳播機制了,也就是 Propagation.REQUIRED。

國際慣例,列出事務傳播機制:

1、PROPAGATION_REQUIRED
當前沒事務,則創建事務;存在事務,就加入該事務,這是最常用的設置。

2、PROPAGATION_SUPPORTS
當前存在事務,就加入事務,當前不存在事務,就以非事務方式執行。

3、PROPAGATION_MANDATORY
當前存在事務,就加入事務;當前不存在事務,就拋出異常。

4、PROPAGATION_REQUIRES_NEW
無條件創建新事務。

5、PROPAGATION_NOT_SUPPORTED
以非事務方式執行,如果當前存在事務,就將當前事務掛起。

6、PROPAGATION_NEVER
以非事務方式運行,如果存在事務,就拋出異常。

7、PROPAGATION_NESTED
開始執行事務前,先保存一個savepoint,當發生異常時,就回滾到savepoint;沒有異常時,跟著外部事務一起提交或回滾。

具體原因

1、看了上邊的事務傳播機制,繼續細化問題,內外層共用一個事務,內層拋出異常,會導致整個事務失敗。

2、繼續分析,外層邏輯進行了 try catch,就導致內層的異常無法繼續向上拋出,外層事務會繼續提交。

3、事務提交時,進行事務狀態的判斷,就發現這個事務是失敗的,需要回滾,所以拋出了 Transaction silently rolled back because it has been marked as rollback-only 的異常。

怎么解決?

根據業務場景選擇合適的方案。

1、當前這種場景,直接把外層邏輯中的 try catch 去掉即可。異常直接向上拋,事務就不會繼續提交,調用方拿到的就是一手的異常;

2、如果內層不是核心邏輯,記錄個日志啥的,可以把內層事務配置為 @Transactional(rollbackFor = Throwable.class, propagation = Propagation.REQUIRES_NEW), 無論如何,都創建新的事務,外層事務不受內層事務影響。但是有個問題,外層事務失敗了,內層事務還是把記錄入庫了,有可能產生臟數據;

3、如果外層事務失敗了,內層事務也不能提交,那就可以使用 @Transactional(rollbackFor = Throwable.class, propagation = Propagation.NESTED)。注意:hibernate/jpa 不支持嵌套事務 NESTED,可用 JdbcTemplate 代替。

最后,下面這位粉絲總結的事務不生效問題,大家牢記。面試中能全說出來,Offer基本穩了。

看完上述內容,你們對遇到Spring雙層事務不回滾怎么辦有進一步的了解嗎?如果還想了解更多知識或者相關內容,請關注億速云行業資訊頻道,感謝大家的支持。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

亚洲午夜精品一区二区_中文无码日韩欧免_久久香蕉精品视频_欧美主播一区二区三区美女