Golang 使用事務(wù)的簡(jiǎn)單實(shí)踐
在實(shí)際業(yè)務(wù)開(kāi)發(fā)中,事務(wù)(Transaction)是保證數(shù)據(jù)一致性的重要手段。比如:
- 用戶注冊(cè)時(shí),需要同時(shí)寫(xiě)入用戶表和日志表;
- 訂單支付時(shí),需要同時(shí)扣減庫(kù)存和生成支付流水;
- 轉(zhuǎn)賬時(shí),需要同時(shí)扣減賬戶 A 的余額并增加賬戶 B 的余額。
這些操作必須 要么全部成功,要么全部失敗,否則就會(huì)導(dǎo)致數(shù)據(jù)不一致。本文將結(jié)合 Golang 的示例代碼,介紹如何在項(xiàng)目中優(yōu)雅地使用事務(wù)。
一、事務(wù)的基本概念
事務(wù)具備 ACID 四大特性:
- A(Atomicity,原子性):事務(wù)中的操作要么全部成功,要么全部失敗。
- C(Consistency,一致性):事務(wù)執(zhí)行前后,數(shù)據(jù)必須保持一致。
- I(Isolation,隔離性):多個(gè)事務(wù)之間相互獨(dú)立,互不干擾。
- D(Durability,持久性):事務(wù)一旦提交,數(shù)據(jù)就會(huì)被永久保存。
二、事務(wù)的使用示例
// 需要使用事務(wù)的方法
func (s *userService) funcName(ctx context.Context, req *v1.Req) (*v1.RespData, error) {
// 獲取事務(wù)的最終結(jié)果
err := s.tm.Transaction(ctx, func(ctx context.Context) error {
// 內(nèi)部寫(xiě)相關(guān)的原子性數(shù)據(jù)庫(kù)操作
// 如果任意操作報(bào)錯(cuò),將觸發(fā)回滾,恢復(fù)之前的狀態(tài)
// 調(diào)用數(shù)據(jù)層方法 repository
// repository.CreateUser(ctx, req)
// repository.CreateLog(ctx, req)
// 所有操作均無(wú)錯(cuò)誤,正常退出
return nil
})
// 如果事務(wù)中存在錯(cuò)誤,所有操作都會(huì)被回滾
if err != nil {
return nil, err
}
// 沒(méi)有觸發(fā)事務(wù)報(bào)錯(cuò),正常返回結(jié)果
return &v1.RespData{}, nil
}
三、結(jié)合 GORM 使用事務(wù)
如果你使用的是 GORM,事務(wù)的寫(xiě)法會(huì)更簡(jiǎn)潔:
func (s *userService) CreateOrder(ctx context.Context, req *v1.OrderReq) error {
return s.db.Transaction(func(tx *gorm.DB) error {
// 創(chuàng)建訂單
if err := tx.Create(&Order{UserID: req.UserID, Amount: req.Amount}).Error; err != nil {
return err // 返回錯(cuò)誤會(huì)觸發(fā)回滾
}
// 扣減庫(kù)存
if err := tx.Model(&Product{}).
Where("id = ? AND stock >= ?", req.ProductID, req.Quantity).
Update("stock", gorm.Expr("stock - ?", req.Quantity)).Error; err != nil {
return err
}
// 寫(xiě)入日志
if err := tx.Create(&Log{Action: "create_order", UserID: req.UserID}).Error; err != nil {
return err
}
// 所有操作成功,事務(wù)提交
return nil
})
}
四、事務(wù)的應(yīng)用場(chǎng)景
- 用戶注冊(cè):寫(xiě)入用戶表 + 寫(xiě)入用戶詳情表 + 寫(xiě)入日志表。
- 訂單支付:扣減庫(kù)存 + 生成訂單記錄 + 寫(xiě)入支付流水。
- 資金轉(zhuǎn)賬:賬戶 A 扣款 + 賬戶 B 加款 + 生成轉(zhuǎn)賬記錄。
五、最佳實(shí)踐
- 事務(wù)粒度要小:只包含必要的數(shù)據(jù)庫(kù)操作,避免長(zhǎng)時(shí)間占用連接。
- 錯(cuò)誤處理要及時(shí):一旦事務(wù)中出現(xiàn)錯(cuò)誤,應(yīng)立即返回,觸發(fā)回滾。
- 避免耗時(shí)操作:不要在事務(wù)中調(diào)用外部 API 或執(zhí)行復(fù)雜計(jì)算。
- 封裝事務(wù)邏輯:在服務(wù)層統(tǒng)一封裝事務(wù),減少重復(fù)代碼。
- 結(jié)合 Context:在事務(wù)中傳遞
context.Context,方便控制超時(shí)和取消。
六、常見(jiàn)問(wèn)題 FAQ
Q1:事務(wù)中如何傳遞上下文(Context)?
在事務(wù)回調(diào)函數(shù)中繼續(xù)傳遞 ctx,保證日志、超時(shí)控制等功能生效。例如:
s.db.WithContext(ctx).Transaction(func(tx *gorm.DB) error {
return tx.Create(&User{Name: "Tom"}).Error
})
Q2:如何在事務(wù)中調(diào)用多個(gè) repository?
只需將事務(wù)對(duì)象 tx 傳遞給 repository 方法即可:
func (r *UserRepo) Create(ctx context.Context, tx *gorm.DB, user *User) error {
return tx.WithContext(ctx).Create(user).Error
}
這樣可以保證所有 repository 操作都在同一個(gè)事務(wù)中。
Q3:事務(wù)中能否執(zhí)行外部 API 調(diào)用?
不推薦。外部 API 調(diào)用可能耗時(shí)較長(zhǎng),導(dǎo)致事務(wù)長(zhǎng)時(shí)間占用數(shù)據(jù)庫(kù)連接,影響性能。建議先執(zhí)行事務(wù),再調(diào)用外部服務(wù),或通過(guò)消息隊(duì)列解耦。
Q4:如何處理事務(wù)嵌套?
GORM 默認(rèn)不支持真正的嵌套事務(wù),但可以使用 SavePoint 和 RollbackTo 來(lái)模擬:
tx.SavePoint("sp1")
// ...
tx.RollbackTo("sp1")
七、總結(jié)
事務(wù)是保證數(shù)據(jù)一致性的重要手段。在 Golang 項(xiàng)目中,我們可以通過(guò)事務(wù)管理器或 GORM 的 db.Transaction 來(lái)簡(jiǎn)化事務(wù)的使用。
只要遵循 小粒度、快執(zhí)行、及時(shí)回滾 的原則,就能在項(xiàng)目中高效、安全地使用事務(wù)。
到此這篇關(guān)于Golang 使用事務(wù)的簡(jiǎn)單實(shí)踐的文章就介紹到這了,更多相關(guān)Golang 事務(wù)內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Go標(biāo)準(zhǔn)庫(kù)encoding/gob的具體使用
Go標(biāo)準(zhǔn)庫(kù)encoding/gob實(shí)現(xiàn)二進(jìn)制序列化與反序列化,本文主要介紹了Go標(biāo)準(zhǔn)庫(kù)encoding/gob的具體使用,感興趣的可以了解一下2025-06-06
Golang通道阻塞情況與通道無(wú)阻塞實(shí)現(xiàn)小結(jié)
本文主要介紹了Golang通道阻塞情況與通道無(wú)阻塞實(shí)現(xiàn)小結(jié),詳細(xì)解析了通道的類(lèi)型、操作方法以及垃圾回收機(jī)制,從基礎(chǔ)概念到高級(jí)應(yīng)用,具有一定的參考價(jià)值,感興趣的可以了解一下2024-03-03
Golang創(chuàng)建構(gòu)造函數(shù)的方法超詳細(xì)講解
構(gòu)造器一般面向?qū)ο笳Z(yǔ)言的典型特性,用于初始化變量。Go語(yǔ)言沒(méi)有任何具體構(gòu)造器,但我們能使用該特性去初始化變量。本文介紹不同類(lèi)型構(gòu)造器的差異及其應(yīng)用場(chǎng)景2023-01-01

