mirror of
https://github.com/TeaOSLab/EdgeAPI.git
synced 2025-11-06 18:10:25 +08:00
优化自增锁性能
This commit is contained in:
@@ -1,12 +1,14 @@
|
|||||||
package models
|
package models
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"errors"
|
||||||
"github.com/TeaOSLab/EdgeAPI/internal/zero"
|
"github.com/TeaOSLab/EdgeAPI/internal/zero"
|
||||||
_ "github.com/go-sql-driver/mysql"
|
_ "github.com/go-sql-driver/mysql"
|
||||||
"github.com/iwind/TeaGo/Tea"
|
"github.com/iwind/TeaGo/Tea"
|
||||||
"github.com/iwind/TeaGo/dbs"
|
"github.com/iwind/TeaGo/dbs"
|
||||||
"github.com/iwind/TeaGo/maps"
|
"github.com/iwind/TeaGo/maps"
|
||||||
"github.com/iwind/TeaGo/types"
|
"github.com/iwind/TeaGo/types"
|
||||||
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -119,6 +121,11 @@ func (this *SysLockerDAO) Unlock(tx *dbs.Tx, key string) error {
|
|||||||
|
|
||||||
// Increase 增加版本号
|
// Increase 增加版本号
|
||||||
func (this *SysLockerDAO) Increase(tx *dbs.Tx, key string, defaultValue int64) (int64, error) {
|
func (this *SysLockerDAO) Increase(tx *dbs.Tx, key string, defaultValue int64) (int64, error) {
|
||||||
|
// validate key
|
||||||
|
if strings.Contains(key, "'") {
|
||||||
|
return 0, errors.New("invalid key '" + key + "'")
|
||||||
|
}
|
||||||
|
|
||||||
if tx == nil {
|
if tx == nil {
|
||||||
var result int64
|
var result int64
|
||||||
var err error
|
var err error
|
||||||
@@ -137,7 +144,21 @@ func (this *SysLockerDAO) Increase(tx *dbs.Tx, key string, defaultValue int64) (
|
|||||||
})
|
})
|
||||||
return result, err
|
return result, err
|
||||||
}
|
}
|
||||||
err := this.Query(tx).
|
|
||||||
|
// combine statements to make increasing faster
|
||||||
|
colValue, err := tx.FindCol(0, "INSERT INTO `edgeSysLockers` (`key`, `version`) VALUES ('"+key+"', "+types.String(defaultValue)+") ON DUPLICATE KEY UPDATE `version`=`version`+1; SELECT `version` FROM edgeSysLockers WHERE `key`='"+key+"'")
|
||||||
|
if err != nil {
|
||||||
|
if CheckSQLErrCode(err, 1064 /** syntax error **/) {
|
||||||
|
// continue to use seperated query
|
||||||
|
err = nil
|
||||||
|
} else {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
return types.Int64(colValue), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
err = this.Query(tx).
|
||||||
Reuse(false). // no need to prepare statement in every transaction
|
Reuse(false). // no need to prepare statement in every transaction
|
||||||
InsertOrUpdateQuickly(maps.Map{
|
InsertOrUpdateQuickly(maps.Map{
|
||||||
"key": key,
|
"key": key,
|
||||||
|
|||||||
@@ -12,14 +12,16 @@ import (
|
|||||||
func TestSysLockerDAO_Lock(t *testing.T) {
|
func TestSysLockerDAO_Lock(t *testing.T) {
|
||||||
var tx *dbs.Tx
|
var tx *dbs.Tx
|
||||||
|
|
||||||
isOk, err := SharedSysLockerDAO.Lock(tx, "test", 600)
|
var dao = NewSysLockerDAO()
|
||||||
|
|
||||||
|
isOk, err := dao.Lock(tx, "test", 600)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
t.Log(isOk)
|
t.Log(isOk)
|
||||||
|
|
||||||
if isOk {
|
if isOk {
|
||||||
err = SharedSysLockerDAO.Unlock(tx, "test")
|
err = dao.Unlock(tx, "test")
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
@@ -128,3 +130,19 @@ func TestSysLocker_Increase_Performance(t *testing.T) {
|
|||||||
|
|
||||||
t.Log("cost:", time.Since(before).Seconds()*1000, "ms")
|
t.Log("cost:", time.Since(before).Seconds()*1000, "ms")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func BenchmarkSysLockerDAO_Increase(b *testing.B) {
|
||||||
|
var dao = NewSysLockerDAO()
|
||||||
|
_, _ = dao.Increase(nil, "hello", 0)
|
||||||
|
|
||||||
|
b.ResetTimer()
|
||||||
|
|
||||||
|
b.RunParallel(func(pb *testing.PB) {
|
||||||
|
for pb.Next() {
|
||||||
|
_, err := dao.Increase(nil, "hello", 0)
|
||||||
|
if err != nil {
|
||||||
|
b.Fatal(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|||||||
@@ -80,3 +80,12 @@ func CheckSQLDuplicateErr(err error) bool {
|
|||||||
}
|
}
|
||||||
return CheckSQLErrCode(err, 1062)
|
return CheckSQLErrCode(err, 1062)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// IsMySQLError Check error is MySQLError
|
||||||
|
func IsMySQLError(err error) bool {
|
||||||
|
if err == nil {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
_, ok := err.(*mysql.MySQLError)
|
||||||
|
return ok
|
||||||
|
}
|
||||||
|
|||||||
@@ -97,7 +97,13 @@ func (this *Setup) Run() error {
|
|||||||
}
|
}
|
||||||
for _, db := range config.DBs {
|
for _, db := range config.DBs {
|
||||||
// 可以同时运行多条语句
|
// 可以同时运行多条语句
|
||||||
|
if !strings.Contains(db.Dsn, "multiStatements=") {
|
||||||
|
if strings.Contains(db.Dsn, "?") {
|
||||||
db.Dsn += "&multiStatements=true"
|
db.Dsn += "&multiStatements=true"
|
||||||
|
} else {
|
||||||
|
db.Dsn += "?multiStatements=true"
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
dbConfig, ok := config.DBs[Tea.Env]
|
dbConfig, ok := config.DBs[Tea.Env]
|
||||||
if !ok {
|
if !ok {
|
||||||
|
|||||||
Reference in New Issue
Block a user