-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathredisLock.go
99 lines (84 loc) · 2.55 KB
/
redisLock.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
package redis
import (
"context"
"time"
"github.com/farseer-go/fs/core"
"github.com/farseer-go/fs/flog"
)
// 分布式锁
type redisLock struct {
*redisManager
}
type lockResult struct {
key string // 锁名称
val string // 锁值
expiration time.Duration
*redisManager
}
// LockNew 创建锁
func (r redisLock) LockNew(key, val string, expiration time.Duration) core.ILock {
return &lockResult{
key: key,
val: val,
expiration: expiration,
redisManager: r.redisManager,
}
}
// TryLock 尝试加锁
func (receiver *lockResult) TryLock() bool {
traceDetail := receiver.traceManager.TraceRedis("TryLock", receiver.key, "")
result, err := receiver.GetClient().SetNX(context.Background(), receiver.key, receiver.val, receiver.expiration).Result()
defer func() { traceDetail.End(err) }()
if err != nil {
_ = flog.Errorf("redis加锁异常:%s", err.Error())
}
return result
}
// TryLockRun 尝试加锁,执行完后,自动释放锁(未获取到时直接退出)
func (receiver *lockResult) TryLockRun(fn func()) bool {
traceDetail := receiver.traceManager.TraceRedis("TryLockRun", receiver.key, "")
//sw := stopwatch.StartNew()
result, err := receiver.GetClient().SetNX(context.Background(), receiver.key, receiver.val, receiver.expiration).Result()
defer func() { traceDetail.End(err) }()
//flog.Debugf("获取Redis锁,耗时:%s", sw.GetMicrosecondsText())
if err != nil {
_ = flog.Errorf("redis加锁异常:%s", err.Error())
}
if result {
defer receiver.ReleaseLock()
fn()
}
return result
}
// GetLock 获取锁,直到获取成功
func (receiver *lockResult) GetLock() {
traceDetail := receiver.traceManager.TraceRedis("GetLock", receiver.key, "")
var err error
defer func() { traceDetail.End(err) }()
for {
var result bool
result, err = receiver.GetClient().SetNX(context.Background(), receiver.key, receiver.val, receiver.expiration).Result()
if err != nil {
_ = flog.Errorf("redis加锁异常:%s", err.Error())
}
if result {
return
}
time.Sleep(100 * time.Millisecond)
}
}
// GetLockRun 获取锁,直到获取成功,执行完后,自动释放锁
func (receiver *lockResult) GetLockRun(fn func()) {
for {
if receiver.TryLockRun(fn) {
return
}
time.Sleep(100 * time.Millisecond)
}
}
// ReleaseLock 锁放锁
func (receiver *lockResult) ReleaseLock() {
traceDetail := receiver.traceManager.TraceRedis("ReleaseLock", receiver.key, "")
_, err := receiver.GetClient().Del(context.Background(), receiver.key).Result()
defer func() { traceDetail.End(err) }()
}