gf/os/gmlock/gmlock_z_unit_test.go
2023-05-25 21:58:11 +08:00

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)
})
}