欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Seata分布式事務(wù)出現(xiàn)ABA問題解決

 更新時間:2022年11月09日 16:26:13   作者:夢想實現(xiàn)家_Z  
這篇文章主要為大家介紹了Seata分布式事務(wù)出現(xiàn)ABA問題解決方法示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪

前言

兄弟們,最近處理了一個seata的issue,關(guān)于seata分布式事務(wù)長期回滾失敗后,突然回滾成功了:

這個問題的出現(xiàn)需要以下兩個契機:

  • 在執(zhí)行分布式事務(wù)期間,有本地事務(wù)與分布式事務(wù)操作同一張表中的數(shù)據(jù)導(dǎo)致臟寫產(chǎn)生;
  • 在回滾時,seata對比afterImage與當(dāng)前數(shù)據(jù)不一致,導(dǎo)致回滾失敗,此時會一直重試;
  • 當(dāng)手工校準(zhǔn)數(shù)據(jù)后,某一時刻afterImage與當(dāng)前數(shù)據(jù)一致,此時回滾重試成功,ABA問題產(chǎn)生;

從源碼中定位原因

為了避免ABA問題的產(chǎn)生,通過與seata社區(qū)的大佬討論,最終決定在回滾時,如果對比afterImage與當(dāng)前數(shù)據(jù)不一致的情況下,不再嘗試回滾重試。這樣的話,即使后續(xù)通過人工校準(zhǔn)后,也不會回滾了。但是這樣有另一個問題,就是人工校準(zhǔn)后,這個分布式事務(wù)就一直遺留在數(shù)據(jù)庫中無法刪除了。針對這個問題,seata應(yīng)該要提供一個restful api讓開發(fā)人員在數(shù)據(jù)校準(zhǔn)后能夠刪除掉對應(yīng)的分布式事務(wù)數(shù)據(jù)。

在seata源碼中,如果校驗afterImage與當(dāng)前數(shù)據(jù)不一致后,會拋出SQLException,最終會被上層代碼捕獲包裝成BranchTransactionException異常,但是里面的code屬性是BranchRollbackFailed_Retriable,這也是導(dǎo)致seata一直重試回滾的根本原因:

Result<Boolean> afterEqualsCurrentResult = DataCompareUtils.isRecordsEquals(afterRecords, currentRecords);
        if (!afterEqualsCurrentResult.getResult()) {
            // 先比較afterImage與當(dāng)前數(shù)據(jù),如果不一致,那么再比較當(dāng)前數(shù)據(jù)和beforeImage是否一致
            Result<Boolean> beforeEqualsCurrentResult = DataCompareUtils.isRecordsEquals(beforeRecords, currentRecords);
            // 如果當(dāng)前數(shù)據(jù)和beforeImage一致,那么不需要回滾了,因為相當(dāng)于已經(jīng)回滾了
            if (beforeEqualsCurrentResult.getResult()) {
                if (LOGGER.isInfoEnabled()) {
                    LOGGER.info("Stop rollback because there is no data change " +
                            "between the before data snapshot and the current data snapshot.");
                }
                // no need continue undo.
                return false;
            } else {
                // 否則,直接拋出SQLException,并告知undo log臟寫了
                if (LOGGER.isInfoEnabled()) {
                    if (StringUtils.isNotBlank(afterEqualsCurrentResult.getErrMsg())) {
                        LOGGER.info(afterEqualsCurrentResult.getErrMsg(), afterEqualsCurrentResult.getErrMsgParams());
                    }
                }
                if (LOGGER.isDebugEnabled()) {
                    LOGGER.debug("check dirty data failed, old and new data are not equal, " +
                            "tableName:[" + sqlUndoLog.getTableName() + "]," +
                            "oldRows:[" + JSON.toJSONString(afterRecords.getRows()) + "]," +
                            "newRows:[" + JSON.toJSONString(currentRecords.getRows()) + "].");
                }
                throw new SQLException("Has dirty records when undo.");
            }
        }

在上層調(diào)用代碼中,我們可以找到這樣一段:

catch (Throwable e) {
    if (conn != null) {
        try {
            conn.rollback();
        } catch (SQLException rollbackEx) {
            LOGGER.warn("Failed to close JDBC resource while undo ... ", rollbackEx);
        }
    }
    // 包裝異常
    throw new BranchTransactionException(BranchRollbackFailed_Retriable, String
                    .format("Branch session rollback failed and try again later xid = %s branchId = %s %s", xid,
                        branchId, e.getMessage()), e);
}

根據(jù)源碼分析,我們發(fā)現(xiàn)在數(shù)據(jù)校驗后拋出的SQLException會被包裝成code屬性為BranchRollbackFailed_RetriableBranchTransactionException異常,這樣會導(dǎo)致seata不斷重試回滾操作。

如何處理

我們需要將這個SQLException調(diào)整為一個更加具體的異常,比如SQLUndoDirtyException這種能夠明確地表示undo log被臟寫的異常,另外我們在上層代碼中同樣需要針對SQLUndoDirtyException做特殊處理,比如包裝成new BranchTransactionException(BranchRollbackFailed_Unretriable)不可重試的狀態(tài)。

先創(chuàng)建自定義的異常:SQLUndoDirtyException

import java.io.Serializable;
import java.sql.SQLException;
/**
 * @author zouwei
 */
class SQLUndoDirtyException extends SQLException implements Serializable {
    private static final long serialVersionUID = -5168905669539637570L;
    SQLUndoDirtyException(String reason) {
        super(reason);
    }
}

調(diào)整SQLExceptionSQLUndoDirtyException:

Result<Boolean> afterEqualsCurrentResult = DataCompareUtils.isRecordsEquals(afterRecords, currentRecords);
        if (!afterEqualsCurrentResult.getResult()) {
            // 先比較afterImage與當(dāng)前數(shù)據(jù),如果不一致,那么再比較當(dāng)前數(shù)據(jù)和beforeImage是否一致
            Result<Boolean> beforeEqualsCurrentResult = DataCompareUtils.isRecordsEquals(beforeRecords, currentRecords);
            // 如果當(dāng)前數(shù)據(jù)和beforeImage一致,那么不需要回滾了,因為相當(dāng)于已經(jīng)回滾了
            if (beforeEqualsCurrentResult.getResult()) {
                if (LOGGER.isInfoEnabled()) {
                    LOGGER.info("Stop rollback because there is no data change " +
                            "between the before data snapshot and the current data snapshot.");
                }
                // no need continue undo.
                return false;
            } else {
                // 否則,直接拋出SQLException,并告知undo log臟寫了
                if (LOGGER.isInfoEnabled()) {
                    if (StringUtils.isNotBlank(afterEqualsCurrentResult.getErrMsg())) {
                        LOGGER.info(afterEqualsCurrentResult.getErrMsg(), afterEqualsCurrentResult.getErrMsgParams());
                    }
                }
                if (LOGGER.isDebugEnabled()) {
                    LOGGER.debug("check dirty data failed, old and new data are not equal, " +
                            "tableName:[" + sqlUndoLog.getTableName() + "]," +
                            "oldRows:[" + JSON.toJSONString(afterRecords.getRows()) + "]," +
                            "newRows:[" + JSON.toJSONString(currentRecords.getRows()) + "].");
                }
                // 替換為具體的SQLUndoDirtyException異常
                throw new SQLUndoDirtyException("Has dirty records when undo.");
            }
        }

這樣的話,我們在上層代碼中,就可以針對性地處理了:

catch (Throwable e) {
    if (conn != null) {
        try {
            conn.rollback();
        } catch (SQLException rollbackEx) {
            LOGGER.warn("Failed to close JDBC resource while undo ... ", rollbackEx);
        }
     }
     // 如果捕捉的異常為SQLUndoDirtyException,那么包裝為BranchRollbackFailed_Unretriable
     if (e instanceof SQLUndoDirtyException) {
         throw new BranchTransactionException(BranchRollbackFailed_Unretriable, String.format(
                        "Branch session rollback failed because of dirty undo log, please delete the relevant undolog after manually calibrating the data. xid = %s branchId = %s",
                        xid, branchId), e);
      }
      throw new BranchTransactionException(BranchRollbackFailed_Retriable,
                    String.format("Branch session rollback failed and try again later xid = %s branchId = %s %s", xid,
                        branchId, e.getMessage()),
                    e);
}

我們在上層調(diào)用代碼中捕捉指定的SQLUndoDirtyException,直接包裝為BranchRollbackFailed_Unretriable狀態(tài)的BranchTransactionException,這樣我們的分布式事務(wù)就不會一直重試回滾操作了。

下一步就需要開發(fā)人員人工介入校準(zhǔn)數(shù)據(jù)后刪除對應(yīng)的undo log,在一系列操作處理完畢后,另外還需要seata tc端提供對應(yīng)的restful api開放對應(yīng)的手工觸發(fā)回滾的操作,以便保證校準(zhǔn)后的分布式事務(wù)正常結(jié)束。

小結(jié)

我們根據(jù)seata使用人員反饋的問題,通過源碼分析找到了造成問題的原因:

  • 開發(fā)人員在使用seata的時候,對于同一張表的操作沒有使用@GlobalTransactional注解覆蓋到,導(dǎo)致了undo log被臟寫;
  • 當(dāng)產(chǎn)生回滾時,在進(jìn)行數(shù)據(jù)校驗時,發(fā)現(xiàn)afterImage與當(dāng)前數(shù)據(jù)不一致進(jìn)而無法正?;貪L,拋出SQLException,最終包裝成BranchRollbackFailed_Retriable異常,導(dǎo)致seata一直重試回滾;
  • 在數(shù)據(jù)校準(zhǔn)后,某一刻的數(shù)據(jù)與afterImage一致,此時seata就回滾成功,形成ABA問題;

該pr將在1.6版本后解決seata分布式事務(wù)一直嘗試回滾的問題,可以避免ABA問題的產(chǎn)生,后續(xù)還需要提供一些其他功能輔助開發(fā)人員回滾數(shù)據(jù)。

以上就是Seata分布式事務(wù)出現(xiàn)ABA問題解決的詳細(xì)內(nèi)容,更多關(guān)于Seata分布式事務(wù)ABA的資料請關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • PowerJob的DesignateServer工作流程源碼解讀

    PowerJob的DesignateServer工作流程源碼解讀

    這篇文章主要介紹了PowerJob的DesignateServer工作流程源碼解讀,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2024-01-01
  • SpringBoot結(jié)合ElasticSearch實現(xiàn)模糊查詢的項目實踐

    SpringBoot結(jié)合ElasticSearch實現(xiàn)模糊查詢的項目實踐

    本文主要介紹了SpringBoot結(jié)合ElasticSearch實現(xiàn)模糊查詢的項目實踐,主要實現(xiàn)模糊查詢、批量CRUD、排序、分頁和高亮功能,具有一定的參考價值,感興趣的可以了解一下
    2024-03-03
  • Springboot?hibernate-validator?6.x快速校驗示例代碼

    Springboot?hibernate-validator?6.x快速校驗示例代碼

    這篇文章主要介紹了Springboot?hibernate-validator?6.x校驗,本文以6.2.1.Final版本為例解決了log4j版本的漏洞問題,通過實例代碼給大家介紹的非常詳細(xì),需要的朋友可以參考下
    2021-12-12
  • SpringCloud Finchley+Spring Boot 2.0 集成Consul的方法示例(1.2版本)

    SpringCloud Finchley+Spring Boot 2.0 集成Consul的方法示例(1.2版本)

    這篇文章主要介紹了SpringCloud Finchley+Spring Boot 2.0 集成Consul的方法示例(1.2版本),小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2018-08-08
  • Java實現(xiàn)簡單的模板渲染

    Java實現(xiàn)簡單的模板渲染

    這篇文章主要為大家詳細(xì)介紹了Java實現(xiàn)簡單的模板渲染的相關(guān)資料,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2017-12-12
  • SpringBoot整合iText7導(dǎo)出PDF及性能優(yōu)化方式

    SpringBoot整合iText7導(dǎo)出PDF及性能優(yōu)化方式

    在SpringBoot項目中整合iText7庫以導(dǎo)出PDF文件,不僅能夠滿足報告生成需求,而且可以處理復(fù)雜的文檔布局與樣式,整合步驟包括添加Maven依賴、編寫PDF生成代碼,性能優(yōu)化方面,建議使用流式處理、緩存樣式與字體、優(yōu)化HTML/CSS結(jié)構(gòu)、采用異步處理
    2024-09-09
  • 全面了解Java中的CAS機制

    全面了解Java中的CAS機制

    下面小編就為大家?guī)硪黄媪私釰ava中的CAS機制。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-10-10
  • PowerJob的HashedWheelTimer工作流程源碼解讀

    PowerJob的HashedWheelTimer工作流程源碼解讀

    這篇文章主要為大家介紹了PowerJob的HashedWheelTimer工作流程源碼解讀,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2024-01-01
  • 詳解Java的readBytes是怎么實現(xiàn)的

    詳解Java的readBytes是怎么實現(xiàn)的

    眾所周知,Java是一門跨平臺語言,針對不同的操作系統(tǒng)有不同的實現(xiàn),下面小編就來從一個非常簡單的api調(diào)用帶大家來看看Java具體是怎么做的吧
    2023-07-07
  • 淺談HashMap在高并發(fā)下的問題

    淺談HashMap在高并發(fā)下的問題

    這篇文章主要介紹了HashMap在高并發(fā)下的問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2021-07-07

最新評論