-
Notifications
You must be signed in to change notification settings - Fork 0
/
zset.go
123 lines (105 loc) · 1.98 KB
/
zset.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
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
package structx
import (
"sync"
rbtree "github.com/sakeven/RbTree"
"golang.org/x/exp/constraints"
)
type Ordered constraints.Ordered
// ZSet
type ZSet[K, S Ordered] struct {
sync.RWMutex
m Map[K, S]
tree *rbtree.Tree[S, K]
}
// NewZSet
func NewZSet[K, S Ordered]() *ZSet[K, S] {
return &ZSet[K, S]{
m: NewMap[K, S](),
tree: rbtree.NewTree[S, K](),
}
}
// Get
func (z *ZSet[K, S]) Get(key K) (S, bool) {
z.RLock()
defer z.RUnlock()
return z.m.Get(key)
}
// Set upsert value by key.
func (z *ZSet[K, S]) Set(key K, score S) {
z.Lock()
z.set(key, score)
z.Unlock()
}
func (z *ZSet[K, S]) set(key K, score S) {
z.deleteNode(score, key)
z.m.Put(key, score)
z.tree.Insert(score, key)
}
// deleteNode
func (z *ZSet[K, S]) deleteNode(score S, key K) bool {
for it := z.tree.FindIt(score); it != nil; it = it.Next() {
if it.Value == key {
z.tree.Delete(it.Key)
return true
}
if it.Key != score {
return false
}
}
return false
}
// Incr
func (z *ZSet[K, S]) Incr(key K, incr S) S {
z.Lock()
score, ok := z.m.Get(key)
if ok {
z.deleteNode(score, key)
}
score += incr
z.m.Put(key, score)
z.tree.Insert(score, key)
z.Unlock()
return score
}
// Delete
func (z *ZSet[K, S]) Delete(key K) (s S, ok bool) {
z.Lock()
score, ok := z.m.Get(key)
if ok {
z.m.Delete(key)
z.deleteNode(score, key)
}
z.Unlock()
return score, ok
}
// Len
func (z *ZSet[K, S]) Len() int {
z.RLock()
defer z.RUnlock()
return z.m.Len()
}
// Iter iterate all elements by scores.
func (z *ZSet[K, S]) Iter(f func(k K, s S) bool) {
z.RLock()
defer z.RUnlock()
for it := z.tree.Iterator(); it != nil; it = it.Next() {
if f(it.Value, it.Key) {
return
}
}
}
// MarshalJSON
func (z *ZSet[K, S]) MarshalJSON() ([]byte, error) {
return z.m.MarshalJSON()
}
// UnmarshalJSON
func (z *ZSet[K, S]) UnmarshalJSON(src []byte) error {
if err := z.m.UnmarshalJSON(src); err != nil {
return err
}
z.m.All(func(k K, s S) bool {
z.tree.Insert(s, k)
return true
})
return nil
}