溫馨提示×

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

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

Go事務(wù)中止時(shí)真的結(jié)束事務(wù)解析了嗎

發(fā)布時(shí)間:2023-04-17 16:05:59 來(lái)源:億速云 閱讀:100 作者:iii 欄目:開(kāi)發(fā)技術(shù)

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

事務(wù)實(shí)踐

服務(wù)端在進(jìn)行和數(shù)據(jù)庫(kù)交互時(shí),對(duì)于一些場(chǎng)景我們可能會(huì)使用事務(wù)來(lái)保證數(shù)據(jù)的冪等性。比如在一個(gè)更新的場(chǎng)景時(shí)基本操作流程時(shí)如下:

  • 開(kāi)啟數(shù)據(jù)庫(kù)事務(wù)

  • 通過(guò) ID 獲取數(shù)據(jù)記錄

  • 確認(rèn)是否可以進(jìn)行更新操作

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

  • 提交事務(wù)

  • 如果遇到錯(cuò)誤,就回滾事務(wù)

在從數(shù)據(jù)庫(kù)中獲取數(shù)據(jù)時(shí),可以通過(guò)鎖行的方式防止其他服務(wù)或者程序也對(duì)這條記錄進(jìn)行操作,比如使用 select ... for update 方式獲取數(shù)據(jù)并鎖定該記錄。以下是簡(jiǎn)單的使用事務(wù)操作數(shù)據(jù)的的方法:

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
}

事務(wù)說(shuō)明

從上面的源碼整體看起來(lái)沒(méi)什么問(wèn)題。在進(jìn)行相關(guān)的操作時(shí)只要正常刪除從db 中刪除數(shù)據(jù)后就完成提交事務(wù),但是如果在期間如果發(fā)生問(wèn)題就會(huì)返回error就會(huì)引發(fā) rollback 操作。

但還有一個(gè)需要注意的點(diǎn),當(dāng)獲取到的數(shù)據(jù)時(shí),判斷到該記錄已經(jīng)被刪除時(shí),就會(huì)結(jié)束操作,但是結(jié)束操作卻沒(méi)有對(duì)事務(wù)進(jìn)行釋放操作,所以就會(huì)造成一個(gè)很大的問(wèn)題:數(shù)據(jù)量大的時(shí)候就會(huì)造成整個(gè)后續(xù)所有的請(qǐng)求都超時(shí),導(dǎo)致所有的請(qǐng)求都不能完成操作。

tx.releaseConn(err)

可以看下事務(wù)實(shí)現(xiàn)的源碼,無(wú)論在 rollback 還是 commit 都會(huì)有 releaseConn 釋放連接,所以之前的例子中 defer 函數(shù)僅在出現(xiàn)錯(cuò)誤的時(shí)才調(diào)用回滾,如果不提交也不回滾就會(huì)導(dǎo)致事務(wù)一直處于活躍的狀態(tài),就會(huì)一直持有該事務(wù),其請(qǐng)求再過(guò)來(lái)時(shí)達(dá)到最大值時(shí)就會(huì)造成事務(wù)超時(shí)。

優(yōu)化方案

解決問(wèn)題有一個(gè)很簡(jiǎn)單的的方案就是每個(gè)判斷 error 的條件下都進(jìn)行回滾。也可以直接在 defer 函數(shù)改成回滾事務(wù),提交事務(wù)后再執(zhí)行回滾也不會(huì)執(zhí)行任何操作。

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

但是沒(méi)有任何更改也進(jìn)行提交,然后只有發(fā)生錯(cuò)誤才進(jìn)行回滾可能會(huì)影響代碼的可讀性。在開(kāi)啟事務(wù)的方法中你會(huì)看到在調(diào)用 beginDC 的方法中有使用 context 服務(wù)上下文進(jìn)行回滾事務(wù)。所以還有一個(gè)方案就是通過(guò)取消上下文來(lái)讓事務(wù)結(jié)束從而釋放鎖。

// 方法 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()

所以我們可以直接使用取消上下文的方法,可以先創(chuàng)建一個(gè)新的取消上下文,如果沒(méi)有回滾或者提交時(shí),最后執(zhí)行cancel 就會(huì)通知事務(wù)已完成,然后就會(huì)關(guān)閉事務(wù)。

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()
        }
    }()
......
}

以上就是關(guān)于“Go事務(wù)中止時(shí)真的結(jié)束事務(wù)解析了嗎”這篇文章的內(nèi)容,相信大家都有了一定的了解,希望小編分享的內(nèi)容對(duì)大家有幫助,若想了解更多相關(guān)的知識(shí)內(nèi)容,請(qǐng)關(guān)注億速云行業(yè)資訊頻道。

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

go
AI