2018-08-30 00:00:15 +08:00
|
|
|
|
// Copyright 2018 gf Author(https://gitee.com/johng/gf). 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://gitee.com/johng/gf.
|
|
|
|
|
|
|
|
|
|
|
|
package gmlock
|
|
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
|
"gitee.com/johng/gf/g/container/gmap"
|
|
|
|
|
|
"gitee.com/johng/gf/g/os/gtime"
|
|
|
|
|
|
"time"
|
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
// 内存锁管理对象
|
|
|
|
|
|
type Locker struct {
|
|
|
|
|
|
m *gmap.StringInterfaceMap
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 创建一把内存锁使用的底层RWLocker
|
|
|
|
|
|
func New() *Locker {
|
|
|
|
|
|
return &Locker{
|
|
|
|
|
|
m : gmap.NewStringInterfaceMap(),
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 内存写锁,如果锁成功返回true,失败则返回false;过期时间单位为毫秒,默认为0表示不过期
|
|
|
|
|
|
func (l *Locker) TryLock(key string, expire...int) bool {
|
|
|
|
|
|
return l.doLock(key, l.getExpire(expire...), true)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 内存写锁,锁成功返回true,失败时阻塞,当失败时表示有其他写锁存在;过期时间单位为毫秒,默认为0表示不过期
|
|
|
|
|
|
func (l *Locker) Lock(key string, expire...int) {
|
|
|
|
|
|
l.doLock(key, l.getExpire(expire...), false)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 解除基于内存锁的写锁
|
|
|
|
|
|
func (l *Locker) Unlock(key string) {
|
|
|
|
|
|
if v := l.m.Get(key); v != nil {
|
|
|
|
|
|
v.(*Mutex).Unlock()
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 内存读锁,如果锁成功返回true,失败则返回false;过期时间单位为毫秒,默认为0表示不过期
|
|
|
|
|
|
func (l *Locker) TryRLock(key string, expire...int) bool {
|
|
|
|
|
|
return l.doRLock(key, l.getExpire(expire...), true)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 内存写锁,锁成功返回true,失败时阻塞,当失败时表示有写锁存在;过期时间单位为毫秒,默认为0表示不过期
|
|
|
|
|
|
func (l *Locker) RLock(key string, expire...int) {
|
|
|
|
|
|
l.doRLock(key, l.getExpire(expire...), false)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 解除基于内存锁的读锁
|
|
|
|
|
|
func (l *Locker) RUnlock(key string) {
|
|
|
|
|
|
if v := l.m.Get(key); v != nil {
|
|
|
|
|
|
v.(*Mutex).RUnlock()
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 获得过期时间,没有设置时默认为0不过期
|
|
|
|
|
|
func (l *Locker) getExpire(expire...int) int {
|
|
|
|
|
|
e := 0
|
|
|
|
|
|
if len(expire) > 0 {
|
|
|
|
|
|
e = expire[0]
|
|
|
|
|
|
}
|
|
|
|
|
|
return e
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 内存写锁,当try==true时,如果锁成功返回true,失败则返回false;try==false时,成功时立即返回,否则阻塞等待
|
|
|
|
|
|
func (l *Locker) doLock(key string, expire int, try bool) bool {
|
|
|
|
|
|
mu := l.getOrNewMutex(key)
|
|
|
|
|
|
ok := true
|
|
|
|
|
|
if try {
|
|
|
|
|
|
ok = mu.TryLock()
|
|
|
|
|
|
} else {
|
|
|
|
|
|
mu.Lock()
|
|
|
|
|
|
}
|
|
|
|
|
|
if ok && expire > 0 {
|
|
|
|
|
|
// 异步goroutine计时处理
|
2018-08-30 23:21:53 +08:00
|
|
|
|
wid := mu.wid.Val()
|
2018-08-30 00:00:15 +08:00
|
|
|
|
gtime.SetTimeout(time.Duration(expire)*time.Millisecond, func() {
|
2018-08-30 23:21:53 +08:00
|
|
|
|
if wid == mu.wid.Val() {
|
|
|
|
|
|
mu.Unlock()
|
|
|
|
|
|
}
|
2018-08-30 00:00:15 +08:00
|
|
|
|
})
|
|
|
|
|
|
}
|
|
|
|
|
|
return ok
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 内存读锁,当try==true时,如果锁成功返回true,失败则返回false;try==false时,成功时立即返回,否则阻塞等待
|
|
|
|
|
|
func (l *Locker) doRLock(key string, expire int, try bool) bool {
|
|
|
|
|
|
mu := l.getOrNewMutex(key)
|
|
|
|
|
|
ok := true
|
|
|
|
|
|
if try {
|
|
|
|
|
|
ok = mu.TryRLock()
|
|
|
|
|
|
} else {
|
|
|
|
|
|
mu.RLock()
|
|
|
|
|
|
}
|
|
|
|
|
|
if ok && expire > 0 {
|
|
|
|
|
|
// 异步goroutine计时处理
|
2018-08-30 23:21:53 +08:00
|
|
|
|
rid := mu.rid.Val()
|
2018-08-30 00:00:15 +08:00
|
|
|
|
gtime.SetTimeout(time.Duration(expire)*time.Millisecond, func() {
|
2018-08-30 23:21:53 +08:00
|
|
|
|
if rid == mu.rid.Val() {
|
|
|
|
|
|
mu.RUnlock()
|
|
|
|
|
|
}
|
2018-08-30 00:00:15 +08:00
|
|
|
|
})
|
|
|
|
|
|
}
|
|
|
|
|
|
return ok
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 根据指定key查询或者创建新的Mutex
|
|
|
|
|
|
func (l *Locker) getOrNewMutex(key string) (mu *Mutex) {
|
|
|
|
|
|
// 优先进行读取检查,提高读取效率
|
|
|
|
|
|
if v := l.m.Get(key); v != nil {
|
|
|
|
|
|
mu = v.(*Mutex)
|
|
|
|
|
|
}
|
|
|
|
|
|
if mu == nil {
|
|
|
|
|
|
l.m.LockFunc(func(m map[string]interface{}) {
|
|
|
|
|
|
// 这里必须再进行一次查询,上面那一次使用的是读锁,支持并发效率高;这里面的是写锁,只支持1个goroutine操作
|
|
|
|
|
|
if v, ok := m[key]; ok {
|
|
|
|
|
|
mu = v.(*Mutex)
|
|
|
|
|
|
}
|
|
|
|
|
|
if mu == nil {
|
|
|
|
|
|
mu = NewMutex()
|
|
|
|
|
|
m[key] = mu
|
|
|
|
|
|
}
|
|
|
|
|
|
})
|
|
|
|
|
|
}
|
|
|
|
|
|
return
|
|
|
|
|
|
}
|