mirror of
https://github.com/TeaOSLab/EdgeAPI.git
synced 2025-11-03 15:00:27 +08:00
优化自增锁算法
This commit is contained in:
@@ -1,6 +1,7 @@
|
|||||||
package models
|
package models
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"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"
|
||||||
@@ -11,6 +12,10 @@ import (
|
|||||||
|
|
||||||
type SysLockerDAO dbs.DAO
|
type SysLockerDAO dbs.DAO
|
||||||
|
|
||||||
|
// concurrent transactions control
|
||||||
|
// 考虑到存在多个API节点的可能性,容量不能太大,也不能使用mutex
|
||||||
|
var sysLockerConcurrentLimiter = make(chan zero.Zero, 8)
|
||||||
|
|
||||||
func NewSysLockerDAO() *SysLockerDAO {
|
func NewSysLockerDAO() *SysLockerDAO {
|
||||||
return dbs.NewDAO(&SysLockerDAO{
|
return dbs.NewDAO(&SysLockerDAO{
|
||||||
DAOObject: dbs.DAOObject{
|
DAOObject: dbs.DAOObject{
|
||||||
@@ -117,6 +122,12 @@ func (this *SysLockerDAO) Increase(tx *dbs.Tx, key string, defaultValue int64) (
|
|||||||
if tx == nil {
|
if tx == nil {
|
||||||
var result int64
|
var result int64
|
||||||
var err error
|
var err error
|
||||||
|
|
||||||
|
sysLockerConcurrentLimiter <- zero.Zero{} // push
|
||||||
|
defer func() {
|
||||||
|
<-sysLockerConcurrentLimiter // pop
|
||||||
|
}()
|
||||||
|
|
||||||
err = this.Instance.RunTx(func(tx *dbs.Tx) error {
|
err = this.Instance.RunTx(func(tx *dbs.Tx) error {
|
||||||
result, err = this.Increase(tx, key, defaultValue)
|
result, err = this.Increase(tx, key, defaultValue)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@@ -142,11 +153,10 @@ func (this *SysLockerDAO) Increase(tx *dbs.Tx, key string, defaultValue int64) (
|
|||||||
FindInt64Col(0)
|
FindInt64Col(0)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// 读取当前版本号
|
// 读取当前版本号
|
||||||
func (this *SysLockerDAO) Read(tx *dbs.Tx, key string) (int64, error) {
|
func (this *SysLockerDAO) Read(tx *dbs.Tx, key string) (int64, error) {
|
||||||
return this.Query(tx).
|
return this.Query(tx).
|
||||||
Attr("key", key).
|
Attr("key", key).
|
||||||
Result("version").
|
Result("version").
|
||||||
FindInt64Col(0)
|
FindInt64Col(0)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,8 +3,10 @@ package models
|
|||||||
import (
|
import (
|
||||||
_ "github.com/go-sql-driver/mysql"
|
_ "github.com/go-sql-driver/mysql"
|
||||||
"github.com/iwind/TeaGo/dbs"
|
"github.com/iwind/TeaGo/dbs"
|
||||||
|
"github.com/iwind/TeaGo/types"
|
||||||
"sync"
|
"sync"
|
||||||
"testing"
|
"testing"
|
||||||
|
"time"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestSysLockerDAO_Lock(t *testing.T) {
|
func TestSysLockerDAO_Lock(t *testing.T) {
|
||||||
@@ -25,22 +27,89 @@ func TestSysLockerDAO_Lock(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestSysLocker_Increase(t *testing.T) {
|
func TestSysLocker_Increase(t *testing.T) {
|
||||||
count := 100
|
dbs.NotifyReady()
|
||||||
wg := sync.WaitGroup{}
|
|
||||||
|
var count = 1000
|
||||||
|
|
||||||
|
var dao = NewSysLockerDAO()
|
||||||
|
value, err := dao.Read(nil, "hello")
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
t.Log("before", value)
|
||||||
|
|
||||||
|
var locker = sync.Mutex{}
|
||||||
|
var allValueMap = map[int64]bool{}
|
||||||
|
|
||||||
|
var before = time.Now()
|
||||||
|
|
||||||
|
var wg = sync.WaitGroup{}
|
||||||
wg.Add(count)
|
wg.Add(count)
|
||||||
|
|
||||||
for i := 0; i < count; i++ {
|
for i := 0; i < count; i++ {
|
||||||
go func() {
|
go func(i int) {
|
||||||
defer wg.Done()
|
defer wg.Done()
|
||||||
v, err := NewSysLockerDAO().Increase(nil, "hello", 0)
|
|
||||||
|
var key = "hello"
|
||||||
|
v, err := dao.Increase(nil, key, 0)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Log("err:", err)
|
t.Log("err:", err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
t.Log("v:", v)
|
|
||||||
}()
|
locker.Lock()
|
||||||
|
if allValueMap[v] {
|
||||||
|
t.Log("duplicated:", v)
|
||||||
|
} else {
|
||||||
|
allValueMap[v] = true
|
||||||
|
}
|
||||||
|
locker.Unlock()
|
||||||
|
|
||||||
|
//t.Log("v:", v)
|
||||||
|
_ = v
|
||||||
|
}(i)
|
||||||
}
|
}
|
||||||
|
|
||||||
wg.Wait()
|
wg.Wait()
|
||||||
t.Log("ok")
|
|
||||||
|
t.Log("cost:", time.Since(before).Seconds()*1000, "ms")
|
||||||
|
|
||||||
|
value, err = dao.Read(nil, "hello")
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
t.Log("after", value, "values:", len(allValueMap))
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestSysLocker_Increase_Performance(t *testing.T) {
|
||||||
|
dbs.NotifyReady()
|
||||||
|
|
||||||
|
var count = 1000
|
||||||
|
|
||||||
|
var dao = NewSysLockerDAO()
|
||||||
|
|
||||||
|
var before = time.Now()
|
||||||
|
|
||||||
|
var wg = sync.WaitGroup{}
|
||||||
|
wg.Add(count)
|
||||||
|
|
||||||
|
for i := 0; i < count; i++ {
|
||||||
|
go func(i int) {
|
||||||
|
defer wg.Done()
|
||||||
|
|
||||||
|
var key = "hello" + types.String(i%10)
|
||||||
|
v, err := dao.Increase(nil, key, 0)
|
||||||
|
if err != nil {
|
||||||
|
t.Log("err:", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
//t.Log("v:", v)
|
||||||
|
_ = v
|
||||||
|
}(i)
|
||||||
|
}
|
||||||
|
|
||||||
|
wg.Wait()
|
||||||
|
|
||||||
|
t.Log("cost:", time.Since(before).Seconds()*1000, "ms")
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user