gorm樂觀鎖使用小結(jié)
前言
樂觀鎖,顧名思義,就是保持樂觀態(tài)度,在數(shù)據(jù)并發(fā)過程中,不會加鎖,而是在數(shù)據(jù)提交之后,才會檢查沖突,這通過在數(shù)據(jù)表中增加一個版本號(version)字段來實現(xiàn)。如果數(shù)據(jù)在事務處理期間未被其他事務修改,那么版本號就不會發(fā)生變化,事務可以安全提交。如果版本號發(fā)生變化,說明有沖突發(fā)生,事務需要回滾或重新嘗試
grom樂觀鎖機制
gorm樂觀鎖依賴安裝
Gorm是go中一個優(yōu)秀持久化框架,也提供了樂觀鎖機制,通過以下命令安裝依賴
go get -u gorm.io/plugin/optimisticlock
gorm樂觀鎖使用
在定義實體類過程中,通過加上一個Version版本號控制
import "gorm.io/plugin/optimisticlock"
type User struct {
Id int64 `gorm:"primary_key;type:bigint(20);not null;column:id;comment:'主鍵id';" json:"id"`
UserName string `gorm:"column:userName"`
Sex int `gorm:"column:sex"`
Version optimisticlock.Version
CreateTime time.Time `gorm:"column:create_time;datetime(3);autoUpdateTime" json:"createTime"` // 創(chuàng)建時間
UpdatedTime time.Time `gorm:"column:update_time;datetime(3);autoUpdateTime" json:"updateTime"` // 更新時間
}
創(chuàng)建一個user表
SET NAMES utf8mb4; SET FOREIGN_KEY_CHECKS = 0; -- ---------------------------- -- Table structure for user -- ---------------------------- DROP TABLE IF EXISTS `user`; CREATE TABLE `user` ( `id` bigint NOT NULL AUTO_INCREMENT COMMENT '\'主鍵id\'', `userName` longtext CHARACTER SET utf8mb4 COLLATE utf8mb4_bin NULL, `sex` bigint NULL DEFAULT NULL, `version` bigint NULL DEFAULT NULL, `create_time` datetime(3) NULL DEFAULT NULL, `update_time` datetime(3) NULL DEFAULT NULL, PRIMARY KEY (`id`) USING BTREE ) ENGINE = InnoDB AUTO_INCREMENT = 2 CHARACTER SET = utf8mb4 COLLATE = utf8mb4_bin ROW_FORMAT = Dynamic; SET FOREIGN_KEY_CHECKS = 1;
插入數(shù)據(jù)
插入一條id為2的數(shù)據(jù)
package main
import (
"gorm.io/driver/mysql"
"gorm.io/gorm"
"gorm.io/gorm/schema"
"log"
"time"
)
import "gorm.io/plugin/optimisticlock"
type User struct {
Id int64 `gorm:"primary_key;type:bigint(20);not null;column:id;comment:'主鍵id';" json:"id"`
UserName string `gorm:"column:userName"`
Sex int `gorm:"column:sex"`
Version optimisticlock.Version
CreateTime time.Time `gorm:"column:create_time;datetime(3);autoUpdateTime" json:"createTime"` // 創(chuàng)建時間
UpdatedTime time.Time `gorm:"column:update_time;datetime(3);autoUpdateTime" json:"updateTime"` // 更新時間
}
func main() {
dsn := "root:root@tcp(127.0.0.1:3306)/test?charset=utf8mb4&parseTime=True&loc=Local"
db, err := gorm.Open(mysql.Open(dsn), &gorm.Config{
NamingStrategy: schema.NamingStrategy{
SingularTable: true, // 使用單數(shù)表名
},
})
if err != nil {
log.Println("連接數(shù)據(jù)庫錯誤:", err)
return
}
var user = User{
UserName: "aaa",
Sex: 1,
}
result := db.Table("user").Create(&user)
log.Println(result)
log.Println(user)
//var user1 User
//db.Table("user").First(&user1, "id = ?", 1)
//log.Println(user1)
//user1.UserName = "bbbb"
//result1 := db.Table("user").Save(&user1)
//fmt.Println(result1)
}

這時候版本為1
版本號更新
這時候把id=2的數(shù)據(jù)查出來,進行更新操作,版本號會發(fā)生變化
package main
import (
"fmt"
"gorm.io/driver/mysql"
"gorm.io/gorm"
"gorm.io/gorm/schema"
"log"
"time"
)
import "gorm.io/plugin/optimisticlock"
type User struct {
Id int64 `gorm:"primary_key;type:bigint(20);not null;column:id;comment:'主鍵id';" json:"id"`
UserName string `gorm:"column:userName"`
Sex int `gorm:"column:sex"`
Version optimisticlock.Version
CreateTime time.Time `gorm:"column:create_time;datetime(3);autoUpdateTime" json:"createTime"` // 創(chuàng)建時間
UpdatedTime time.Time `gorm:"column:update_time;datetime(3);autoUpdateTime" json:"updateTime"` // 更新時間
}
func main() {
dsn := "root:root@tcp(127.0.0.1:3306)/test?charset=utf8mb4&parseTime=True&loc=Local"
db, err := gorm.Open(mysql.Open(dsn), &gorm.Config{
NamingStrategy: schema.NamingStrategy{
SingularTable: true, // 使用單數(shù)表名
},
})
if err != nil {
log.Println("連接數(shù)據(jù)庫錯誤:", err)
return
}
var user1 User
db.Table("user").First(&user1, "id = ?", 2)
log.Println(user1)
user1.UserName = "bbbb"
result1 := db.Table("user").Save(&user1)
fmt.Println(result1)
}

這時候可以看到更新成功,版本號變成2
總結(jié)
樂觀鎖機制,可以有效保證在并發(fā)過程修改數(shù)據(jù)過程中的不安全問題,但是后面更新失敗的問題,根據(jù)項目,具體問題具體分析
到此這篇關(guān)于gorm樂觀鎖使用小結(jié)的文章就介紹到這了,更多相關(guān)gorm 樂觀鎖內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Go|使用Options模式和建造者模式創(chuàng)建對象實戰(zhàn)
這篇文章主要介紹了Go使用Options模式和建造者模式創(chuàng)建對象實戰(zhàn)詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-04-04
Golang語言JSON解碼函數(shù)Unmarshal的使用
本文主要介紹了Golang語言JSON解碼函數(shù)Unmarshal的使用,文中通過示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下2022-01-01
Go語言Swagger實現(xiàn)為項目生成 API 文檔
Swagger 是一個基于 OpenAPI 規(guī)范設(shè)計的工具,用于為 RESTful API 生成交互式文檔,下面小編就來介紹一下如何在 Go 項目中集成 Swagger,特別是結(jié)合 Gin 框架生成 API 文檔2025-03-03
Go語言實現(xiàn)二進制與十進制互轉(zhuǎn)的示例代碼
這篇文章主要和大家詳細介紹了Go語言中實現(xiàn)二進制與十進制互相轉(zhuǎn)換的示例代碼,文中的代碼簡潔易懂,感興趣的小伙伴可以跟隨小編一起學習一下2023-05-05
Go如何優(yōu)雅的關(guān)閉goroutine協(xié)程
本文將介紹首先為什么需要主動關(guān)閉goroutine,并介紹如何在Go語言中關(guān)閉goroutine的常見套路,包括傳遞終止信號和協(xié)程內(nèi)部捕捉終止信號,之后,文章列舉了需要主動關(guān)閉協(xié)程運行的常見場景,希望通過本文的介紹,讀者能夠掌握如何在適當?shù)臅r候關(guān)閉goroutine2023-05-05

