一文帶你深入了解Go語(yǔ)言中的事務(wù)
背景
近期看到一篇文章,真的感嘆作者的洞察力,在開(kāi)發(fā)時(shí)有可能就會(huì)犯這樣的錯(cuò)誤,所以一定要多學(xué)習(xí),多實(shí)踐。其問(wèn)題就是你在提交事務(wù)時(shí),如果中間有其他業(yè)務(wù)就取消操作,那么事務(wù)也關(guān)閉了嗎?
事務(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()
}
}()
......
}總結(jié)
所以在使用事務(wù)處理業(yè)務(wù)的時(shí)候,一定要注意業(yè)務(wù)邏輯,如果在業(yè)務(wù)邏輯中出現(xiàn)某些條件場(chǎng)景不進(jìn)行操作數(shù)據(jù)庫(kù)時(shí),結(jié)束這次業(yè)務(wù)處理時(shí)也要記得關(guān)閉事務(wù)。要么就關(guān)閉事務(wù),要么就是無(wú)論有沒(méi)有處理都提交事務(wù)。這可能只是一個(gè)小小的問(wèn)題,但是如果在交易的場(chǎng)景中如果沒(méi)有注意就可能造成很大的問(wèn)題。
到此這篇關(guān)于一文帶你深入了解Go語(yǔ)言中的事務(wù)的文章就介紹到這了,更多相關(guān)Go語(yǔ)言事務(wù)內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
使用Go語(yǔ)言創(chuàng)建WebSocket服務(wù)的實(shí)現(xiàn)示例
這篇文章主要介紹了使用Go語(yǔ)言創(chuàng)建WebSocket服務(wù)的實(shí)現(xiàn)示例,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-03-03
詳解Golang中NewTimer計(jì)時(shí)器的底層實(shí)現(xiàn)原理
本文將主要介紹一下Go語(yǔ)言中的NewTimer,首先展示基于NewTimer創(chuàng)建的定時(shí)器來(lái)實(shí)現(xiàn)超時(shí)控制。接著通過(guò)一系列問(wèn)題的跟進(jìn),展示了NewTimer的底層實(shí)現(xiàn)原理,需要的可以參考一下2023-05-05
Go語(yǔ)言Slice切片底層的實(shí)現(xiàn)
本文主要介紹了Go語(yǔ)言Slice切片底層的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2025-04-04
Go語(yǔ)言開(kāi)發(fā)前后端不分離項(xiàng)目詳解
這篇文章主要為大家介紹了Go語(yǔ)言開(kāi)發(fā)前后端不分離項(xiàng)目詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-11-11
利用go語(yǔ)言編寫(xiě)一個(gè)并發(fā)包
這篇文章主要為大家詳細(xì)介紹了如何利用go語(yǔ)言編寫(xiě)一個(gè)并發(fā)包,適合大部分并發(fā)任務(wù),開(kāi)箱即用,文中的示例代碼講解詳細(xì),有需要的小伙伴可以參考下2023-10-10
使用Go語(yǔ)言實(shí)現(xiàn)找出兩個(gè)大文件中相同的記錄
這篇文章主要為大家詳細(xì)介紹了使用Go語(yǔ)言實(shí)現(xiàn)找出兩個(gè)大文件中相同的記錄的相關(guān)知識(shí),文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2024-10-10
詳解如何使用beego orm在postgres中存儲(chǔ)圖片
這篇文章主要為大家介紹了如何使用beego orm在postgres中存儲(chǔ)圖片詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-04-04

