中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

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

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

Go事務中止時真的結束事務解析了嗎

發布時間:2023-04-17 16:05:59 來源:億速云 閱讀:112 作者:iii 欄目:開發技術

這篇“Go事務中止時真的結束事務解析了嗎”文章的知識點大部分人都不太理解,所以小編給大家總結了以下內容,內容詳細,步驟清晰,具有一定的借鑒價值,希望大家閱讀完這篇文章能有所收獲,下面我們一起來看看這篇“Go事務中止時真的結束事務解析了嗎”文章吧。

事務實踐

服務端在進行和數據庫交互時,對于一些場景我們可能會使用事務來保證數據的冪等性。比如在一個更新的場景時基本操作流程時如下:

  • 開啟數據庫事務

  • 通過 ID 獲取數據記錄

  • 確認是否可以進行更新操作

  • 如果可以更新操作就更新記錄

  • 提交事務

  • 如果遇到錯誤,就回滾事務

在從數據庫中獲取數據時,可以通過鎖行的方式防止其他服務或者程序也對這條記錄進行操作,比如使用 select ... for update 方式獲取數據并鎖定該記錄。以下是簡單的使用事務操作數據的的方法:

func (user *UserResp) DeleteUser(ctx context.Context, id string) error {
	tx, err := user.db.BeginTx(ctx, nil)
	if err != nil {
		return err
	}
	defer func() {
		if err != nil {
			tx.Rollback()
		}
	}()
	result, err := user.handler.getById(id)
	if err != nil {
		return err
	}
	if result.IsDeleted {
		return nil
	}
	if err = user.handler.Delete(id); err != nil {
		return err
	}
	if err = tx.Commit(); err != nil {
		return err
	}
	return nil
}

事務說明

從上面的源碼整體看起來沒什么問題。在進行相關的操作時只要正常刪除從db 中刪除數據后就完成提交事務,但是如果在期間如果發生問題就會返回error就會引發 rollback 操作。

但還有一個需要注意的點,當獲取到的數據時,判斷到該記錄已經被刪除時,就會結束操作,但是結束操作卻沒有對事務進行釋放操作,所以就會造成一個很大的問題:數據量大的時候就會造成整個后續所有的請求都超時,導致所有的請求都不能完成操作。

tx.releaseConn(err)

可以看下事務實現的源碼,無論在 rollback 還是 commit 都會有 releaseConn 釋放連接,所以之前的例子中 defer 函數僅在出現錯誤的時才調用回滾,如果不提交也不回滾就會導致事務一直處于活躍的狀態,就會一直持有該事務,其請求再過來時達到最大值時就會造成事務超時。

優化方案

解決問題有一個很簡單的的方案就是每個判斷 error 的條件下都進行回滾。也可以直接在 defer 函數改成回滾事務,提交事務后再執行回滾也不會執行任何操作。

	defer func() {
			tx.Rollback()
	}()

但是沒有任何更改也進行提交,然后只有發生錯誤才進行回滾可能會影響代碼的可讀性。在開啟事務的方法中你會看到在調用 beginDC 的方法中有使用 context 服務上下文進行回滾事務。所以還有一個方案就是通過取消上下文來讓事務結束從而釋放鎖。

// 方法 beginDC 中的代碼片段
ctx, cancel := context.WithCancel(ctx)
	tx = &Tx{
		db:                 db,
		dc:                 dc,
		releaseConn:        release,
		txi:                txi,
		cancel:             cancel,
		keepConnOnRollback: keepConnOnRollback,
		ctx:                ctx,
	}
	go tx.awaitDone()

所以我們可以直接使用取消上下文的方法,可以先創建一個新的取消上下文,如果沒有回滾或者提交時,最后執行cancel 就會通知事務已完成,然后就會關閉事務。

func (user *UserResp) DeleteUser(ctx context.Context, id string) error {
ctx, cancel := context.WithCancel(ctx)
    defer cancel()
    tx, err := s.db.BeginTxx(ctx, nil)
    if err != nil {
        return nil, err
    }
    defer func() {
        if err != nil {
            tx.Rollback()
        }
    }()
......
}

以上就是關于“Go事務中止時真的結束事務解析了嗎”這篇文章的內容,相信大家都有了一定的了解,希望小編分享的內容對大家有幫助,若想了解更多相關的知識內容,請關注億速云行業資訊頻道。

向AI問一下細節

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

go
AI

利川市| 清水河县| 宁陕县| 汶川县| 玛纳斯县| 乳源| 丽江市| 尼木县| 罗源县| 大庆市| 富蕴县| 七台河市| 香河县| 如东县| 原平市| 德安县| 富顺县| 若羌县| 莫力| 昌图县| 巴林左旗| 盘锦市| 山阴县| 勃利县| 探索| 出国| 工布江达县| 开原市| 武隆县| 洛宁县| 包头市| 达日县| 会东县| 鱼台县| 弥勒县| 枣庄市| 佛坪县| 冕宁县| 胶南市| 醴陵市| 军事|