mirror of
https://gitee.com/johng/gf.git
synced 2024-11-30 03:07:45 +08:00
466 lines
9.9 KiB
Go
466 lines
9.9 KiB
Go
// Copyright GoFrame Author(https://goframe.org). All Rights Reserved.
|
|
//
|
|
// This Source Code Form is subject to the terms of the MIT License.
|
|
// If a copy of the MIT was not distributed with this file,
|
|
// You can obtain one at https://github.com/gogf/gf.
|
|
|
|
package gmlock_test
|
|
|
|
import (
|
|
"sync"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/gogf/gf/v2/container/garray"
|
|
"github.com/gogf/gf/v2/os/gmlock"
|
|
"github.com/gogf/gf/v2/test/gtest"
|
|
)
|
|
|
|
func Test_Locker_Lock(t *testing.T) {
|
|
gtest.C(t, func(t *gtest.T) {
|
|
key := "testLock"
|
|
array := garray.New(true)
|
|
go func() {
|
|
gmlock.Lock(key)
|
|
array.Append(1)
|
|
time.Sleep(300 * time.Millisecond)
|
|
gmlock.Unlock(key)
|
|
}()
|
|
go func() {
|
|
time.Sleep(100 * time.Millisecond)
|
|
gmlock.Lock(key)
|
|
array.Append(1)
|
|
gmlock.Unlock(key)
|
|
}()
|
|
time.Sleep(100 * time.Millisecond)
|
|
t.Assert(array.Len(), 1)
|
|
time.Sleep(100 * time.Millisecond)
|
|
t.Assert(array.Len(), 1)
|
|
time.Sleep(200 * time.Millisecond)
|
|
t.Assert(array.Len(), 2)
|
|
gmlock.Remove(key)
|
|
})
|
|
|
|
gtest.C(t, func(t *gtest.T) {
|
|
key := "testLock"
|
|
array := garray.New(true)
|
|
lock := gmlock.New()
|
|
go func() {
|
|
lock.Lock(key)
|
|
array.Append(1)
|
|
time.Sleep(300 * time.Millisecond)
|
|
lock.Unlock(key)
|
|
}()
|
|
go func() {
|
|
time.Sleep(100 * time.Millisecond)
|
|
lock.Lock(key)
|
|
array.Append(1)
|
|
lock.Unlock(key)
|
|
}()
|
|
time.Sleep(100 * time.Millisecond)
|
|
t.Assert(array.Len(), 1)
|
|
time.Sleep(100 * time.Millisecond)
|
|
t.Assert(array.Len(), 1)
|
|
time.Sleep(200 * time.Millisecond)
|
|
t.Assert(array.Len(), 2)
|
|
lock.Clear()
|
|
})
|
|
|
|
}
|
|
|
|
func Test_Locker_TryLock(t *testing.T) {
|
|
gtest.C(t, func(t *gtest.T) {
|
|
key := "testTryLock"
|
|
array := garray.New(true)
|
|
go func() {
|
|
gmlock.Lock(key)
|
|
array.Append(1)
|
|
time.Sleep(300 * time.Millisecond)
|
|
gmlock.Unlock(key)
|
|
}()
|
|
go func() {
|
|
time.Sleep(150 * time.Millisecond)
|
|
if gmlock.TryLock(key) {
|
|
array.Append(1)
|
|
gmlock.Unlock(key)
|
|
}
|
|
}()
|
|
go func() {
|
|
time.Sleep(400 * time.Millisecond)
|
|
if gmlock.TryLock(key) {
|
|
array.Append(1)
|
|
gmlock.Unlock(key)
|
|
}
|
|
}()
|
|
time.Sleep(100 * time.Millisecond)
|
|
t.Assert(array.Len(), 1)
|
|
time.Sleep(100 * time.Millisecond)
|
|
t.Assert(array.Len(), 1)
|
|
time.Sleep(300 * time.Millisecond)
|
|
t.Assert(array.Len(), 2)
|
|
})
|
|
|
|
}
|
|
|
|
func Test_Locker_LockFunc(t *testing.T) {
|
|
//no expire
|
|
gtest.C(t, func(t *gtest.T) {
|
|
key := "testLockFunc"
|
|
array := garray.New(true)
|
|
go func() {
|
|
gmlock.LockFunc(key, func() {
|
|
array.Append(1)
|
|
time.Sleep(300 * time.Millisecond)
|
|
}) //
|
|
}()
|
|
go func() {
|
|
time.Sleep(100 * time.Millisecond)
|
|
gmlock.LockFunc(key, func() {
|
|
array.Append(1)
|
|
})
|
|
}()
|
|
time.Sleep(100 * time.Millisecond)
|
|
t.Assert(array.Len(), 1)
|
|
time.Sleep(100 * time.Millisecond)
|
|
t.Assert(array.Len(), 1) //
|
|
time.Sleep(200 * time.Millisecond)
|
|
t.Assert(array.Len(), 2)
|
|
})
|
|
}
|
|
|
|
func Test_Locker_TryLockFunc(t *testing.T) {
|
|
//no expire
|
|
gtest.C(t, func(t *gtest.T) {
|
|
key := "testTryLockFunc"
|
|
array := garray.New(true)
|
|
go func() {
|
|
gmlock.TryLockFunc(key, func() {
|
|
array.Append(1)
|
|
time.Sleep(200 * time.Millisecond)
|
|
})
|
|
}()
|
|
go func() {
|
|
time.Sleep(100 * time.Millisecond)
|
|
gmlock.TryLockFunc(key, func() {
|
|
array.Append(1)
|
|
})
|
|
}()
|
|
go func() {
|
|
time.Sleep(300 * time.Millisecond)
|
|
gmlock.TryLockFunc(key, func() {
|
|
array.Append(1)
|
|
})
|
|
}()
|
|
time.Sleep(150 * time.Millisecond)
|
|
t.Assert(array.Len(), 1)
|
|
time.Sleep(400 * time.Millisecond)
|
|
t.Assert(array.Len(), 2)
|
|
})
|
|
}
|
|
|
|
func Test_Multiple_Goroutine(t *testing.T) {
|
|
gtest.C(t, func(t *gtest.T) {
|
|
ch := make(chan struct{})
|
|
num := 1000
|
|
wait := sync.WaitGroup{}
|
|
wait.Add(num)
|
|
for i := 0; i < num; i++ {
|
|
go func() {
|
|
defer wait.Done()
|
|
<-ch
|
|
gmlock.Lock("test")
|
|
defer gmlock.Unlock("test")
|
|
time.Sleep(time.Millisecond)
|
|
}()
|
|
}
|
|
close(ch)
|
|
wait.Wait()
|
|
})
|
|
|
|
gtest.C(t, func(t *gtest.T) {
|
|
ch := make(chan struct{})
|
|
num := 100
|
|
wait := sync.WaitGroup{}
|
|
wait.Add(num * 2)
|
|
for i := 0; i < num; i++ {
|
|
go func() {
|
|
defer wait.Done()
|
|
<-ch
|
|
gmlock.Lock("test")
|
|
defer gmlock.Unlock("test")
|
|
time.Sleep(time.Millisecond)
|
|
}()
|
|
}
|
|
for i := 0; i < num; i++ {
|
|
go func() {
|
|
defer wait.Done()
|
|
<-ch
|
|
gmlock.RLock("test")
|
|
defer gmlock.RUnlock("test")
|
|
time.Sleep(time.Millisecond)
|
|
}()
|
|
}
|
|
close(ch)
|
|
wait.Wait()
|
|
})
|
|
}
|
|
|
|
func Test_Locker_RLock(t *testing.T) {
|
|
// RLock before Lock
|
|
gtest.C(t, func(t *gtest.T) {
|
|
key := "testRLockBeforeLock"
|
|
array := garray.New(true)
|
|
go func() {
|
|
gmlock.RLock(key)
|
|
array.Append(1)
|
|
time.Sleep(200 * time.Millisecond)
|
|
gmlock.RUnlock(key)
|
|
}()
|
|
go func() {
|
|
time.Sleep(100 * time.Millisecond)
|
|
gmlock.Lock(key)
|
|
array.Append(1)
|
|
gmlock.Unlock(key)
|
|
}()
|
|
time.Sleep(100 * time.Millisecond)
|
|
t.Assert(array.Len(), 1)
|
|
time.Sleep(200 * time.Millisecond)
|
|
t.Assert(array.Len(), 2)
|
|
})
|
|
|
|
// Lock before RLock
|
|
gtest.C(t, func(t *gtest.T) {
|
|
key := "testLockBeforeRLock"
|
|
array := garray.New(true)
|
|
go func() {
|
|
gmlock.Lock(key)
|
|
array.Append(1)
|
|
time.Sleep(200 * time.Millisecond)
|
|
gmlock.Unlock(key)
|
|
}()
|
|
go func() {
|
|
time.Sleep(100 * time.Millisecond)
|
|
gmlock.RLock(key)
|
|
array.Append(1)
|
|
gmlock.RUnlock(key)
|
|
}()
|
|
time.Sleep(100 * time.Millisecond)
|
|
t.Assert(array.Len(), 1)
|
|
time.Sleep(200 * time.Millisecond)
|
|
t.Assert(array.Len(), 2)
|
|
})
|
|
|
|
// Lock before RLocks
|
|
gtest.C(t, func(t *gtest.T) {
|
|
key := "testLockBeforeRLocks"
|
|
array := garray.New(true)
|
|
go func() {
|
|
gmlock.Lock(key)
|
|
array.Append(1)
|
|
time.Sleep(300 * time.Millisecond)
|
|
gmlock.Unlock(key)
|
|
}()
|
|
go func() {
|
|
time.Sleep(100 * time.Millisecond)
|
|
gmlock.RLock(key)
|
|
array.Append(1)
|
|
time.Sleep(200 * time.Millisecond)
|
|
gmlock.RUnlock(key)
|
|
}()
|
|
go func() {
|
|
time.Sleep(100 * time.Millisecond)
|
|
gmlock.RLock(key)
|
|
array.Append(1)
|
|
time.Sleep(200 * time.Millisecond)
|
|
gmlock.RUnlock(key)
|
|
}()
|
|
time.Sleep(200 * time.Millisecond)
|
|
t.Assert(array.Len(), 1)
|
|
time.Sleep(200 * time.Millisecond)
|
|
t.Assert(array.Len(), 3)
|
|
})
|
|
}
|
|
|
|
func Test_Locker_TryRLock(t *testing.T) {
|
|
// Lock before TryRLock
|
|
gtest.C(t, func(t *gtest.T) {
|
|
key := "testLockBeforeTryRLock"
|
|
array := garray.New(true)
|
|
go func() {
|
|
gmlock.Lock(key)
|
|
array.Append(1)
|
|
time.Sleep(200 * time.Millisecond)
|
|
gmlock.Unlock(key)
|
|
}()
|
|
go func() {
|
|
time.Sleep(100 * time.Millisecond)
|
|
if gmlock.TryRLock(key) {
|
|
array.Append(1)
|
|
gmlock.RUnlock(key)
|
|
}
|
|
}()
|
|
time.Sleep(150 * time.Millisecond)
|
|
t.Assert(array.Len(), 1)
|
|
time.Sleep(200 * time.Millisecond)
|
|
t.Assert(array.Len(), 1)
|
|
})
|
|
|
|
// Lock before TryRLocks
|
|
gtest.C(t, func(t *gtest.T) {
|
|
key := "testLockBeforeTryRLocks"
|
|
array := garray.New(true)
|
|
go func() {
|
|
gmlock.Lock(key)
|
|
array.Append(1)
|
|
time.Sleep(200 * time.Millisecond)
|
|
gmlock.Unlock(key)
|
|
}()
|
|
go func() {
|
|
time.Sleep(100 * time.Millisecond)
|
|
if gmlock.TryRLock(key) {
|
|
array.Append(1)
|
|
gmlock.RUnlock(key)
|
|
}
|
|
}()
|
|
go func() {
|
|
time.Sleep(300 * time.Millisecond)
|
|
if gmlock.TryRLock(key) {
|
|
array.Append(1)
|
|
gmlock.RUnlock(key)
|
|
}
|
|
}()
|
|
time.Sleep(150 * time.Millisecond)
|
|
t.Assert(array.Len(), 1)
|
|
time.Sleep(200 * time.Millisecond)
|
|
t.Assert(array.Len(), 2)
|
|
})
|
|
}
|
|
|
|
func Test_Locker_RLockFunc(t *testing.T) {
|
|
// RLockFunc before Lock
|
|
gtest.C(t, func(t *gtest.T) {
|
|
key := "testRLockFuncBeforeLock"
|
|
array := garray.New(true)
|
|
go func() {
|
|
gmlock.RLockFunc(key, func() {
|
|
array.Append(1)
|
|
time.Sleep(200 * time.Millisecond)
|
|
})
|
|
}()
|
|
go func() {
|
|
time.Sleep(100 * time.Millisecond)
|
|
gmlock.Lock(key)
|
|
array.Append(1)
|
|
gmlock.Unlock(key)
|
|
}()
|
|
time.Sleep(150 * time.Millisecond)
|
|
t.Assert(array.Len(), 1)
|
|
time.Sleep(200 * time.Millisecond)
|
|
t.Assert(array.Len(), 2)
|
|
})
|
|
|
|
// Lock before RLockFunc
|
|
gtest.C(t, func(t *gtest.T) {
|
|
key := "testLockBeforeRLockFunc"
|
|
array := garray.New(true)
|
|
go func() {
|
|
gmlock.Lock(key)
|
|
array.Append(1)
|
|
time.Sleep(200 * time.Millisecond)
|
|
gmlock.Unlock(key)
|
|
}()
|
|
go func() {
|
|
time.Sleep(100 * time.Millisecond)
|
|
gmlock.RLockFunc(key, func() {
|
|
array.Append(1)
|
|
})
|
|
}()
|
|
time.Sleep(100 * time.Millisecond)
|
|
t.Assert(array.Len(), 1)
|
|
time.Sleep(200 * time.Millisecond)
|
|
t.Assert(array.Len(), 2)
|
|
})
|
|
|
|
// Lock before RLockFuncs
|
|
gtest.C(t, func(t *gtest.T) {
|
|
key := "testLockBeforeRLockFuncs"
|
|
array := garray.New(true)
|
|
go func() {
|
|
gmlock.Lock(key)
|
|
array.Append(1)
|
|
time.Sleep(200 * time.Millisecond)
|
|
gmlock.Unlock(key)
|
|
}()
|
|
go func() {
|
|
time.Sleep(100 * time.Millisecond)
|
|
gmlock.RLockFunc(key, func() {
|
|
array.Append(1)
|
|
time.Sleep(200 * time.Millisecond)
|
|
})
|
|
}()
|
|
go func() {
|
|
time.Sleep(100 * time.Millisecond)
|
|
gmlock.RLockFunc(key, func() {
|
|
array.Append(1)
|
|
time.Sleep(200 * time.Millisecond)
|
|
})
|
|
}()
|
|
time.Sleep(100 * time.Millisecond)
|
|
t.Assert(array.Len(), 1)
|
|
time.Sleep(200 * time.Millisecond)
|
|
t.Assert(array.Len(), 3)
|
|
})
|
|
}
|
|
|
|
func Test_Locker_TryRLockFunc(t *testing.T) {
|
|
// Lock before TryRLockFunc
|
|
gtest.C(t, func(t *gtest.T) {
|
|
key := "testLockBeforeTryRLockFunc"
|
|
array := garray.New(true)
|
|
go func() {
|
|
gmlock.Lock(key)
|
|
array.Append(1)
|
|
time.Sleep(200 * time.Millisecond)
|
|
gmlock.Unlock(key)
|
|
}()
|
|
go func() {
|
|
time.Sleep(100 * time.Millisecond)
|
|
gmlock.TryRLockFunc(key, func() {
|
|
array.Append(1)
|
|
})
|
|
}()
|
|
time.Sleep(100 * time.Millisecond)
|
|
t.Assert(array.Len(), 1)
|
|
time.Sleep(200 * time.Millisecond)
|
|
t.Assert(array.Len(), 1)
|
|
})
|
|
|
|
// Lock before TryRLockFuncs
|
|
gtest.C(t, func(t *gtest.T) {
|
|
key := "testLockBeforeTryRLockFuncs"
|
|
array := garray.New(true)
|
|
go func() {
|
|
gmlock.Lock(key)
|
|
array.Append(1)
|
|
time.Sleep(200 * time.Millisecond)
|
|
gmlock.Unlock(key)
|
|
}()
|
|
go func() {
|
|
time.Sleep(100 * time.Millisecond)
|
|
gmlock.TryRLockFunc(key, func() {
|
|
array.Append(1)
|
|
})
|
|
}()
|
|
go func() {
|
|
time.Sleep(300 * time.Millisecond)
|
|
gmlock.TryRLockFunc(key, func() {
|
|
array.Append(1)
|
|
})
|
|
}()
|
|
time.Sleep(100 * time.Millisecond)
|
|
t.Assert(array.Len(), 1)
|
|
time.Sleep(300 * time.Millisecond)
|
|
t.Assert(array.Len(), 2)
|
|
})
|
|
}
|