-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathcmap.go
126 lines (102 loc) · 2.21 KB
/
cmap.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
124
125
126
package cmap
type GetFunc func(exists bool, value interface{}) (processedValue interface{})
type UpsertFunc func(exists bool, oldValue interface{}) (newValue interface{})
type SetIfFunc func(exists bool, value interface{}) (newValue interface{}, isSetValue bool)
type RemoveIfFunc func(exists bool, value interface{}) bool
type CMap struct {
s *slab
}
func New(funcs ...cmapOptionFunc) *CMap {
opt := newDefaultOption()
for _, fn := range funcs {
fn(opt)
}
return &CMap{
s: newSlab(opt),
}
}
func (c *CMap) Set(key string, value interface{}) {
m := c.s.GetShard(key)
m.Lock()
defer m.Unlock()
m.Set(key, value)
}
func (c *CMap) Get(key string) (interface{}, bool) {
m := c.s.GetShard(key)
m.RLock()
defer m.RUnlock()
return m.Get(key)
}
func (c *CMap) GetRLocked(key string, fn GetFunc) interface{} {
m := c.s.GetShard(key)
m.RLock()
defer m.RUnlock()
v, ok := m.Get(key)
return fn(ok, v)
}
func (c *CMap) Remove(key string) (interface{}, bool) {
m := c.s.GetShard(key)
m.Lock()
defer m.Unlock()
return m.Remove(key)
}
func (c *CMap) Len() int {
count := 0
for _, m := range c.s.Shards() {
m.RLock()
count += m.Len()
m.RUnlock()
}
return count
}
func (c *CMap) Keys() []string {
shards := c.s.Shards()
keys := make([]string, 0, len(shards))
for _, m := range shards {
m.RLock()
keys = append(keys, m.Keys()...)
m.RUnlock()
}
return keys
}
func (c *CMap) Upsert(key string, fn UpsertFunc) (newValue interface{}) {
m := c.s.GetShard(key)
m.Lock()
defer m.Unlock()
oldValue, ok := m.Get(key)
newValue = fn(ok, oldValue)
m.Set(key, newValue)
return
}
func (c *CMap) SetIfAbsent(key string, value interface{}) (updated bool) {
m := c.s.GetShard(key)
m.Lock()
defer m.Unlock()
if _, ok := m.Get(key); ok != true {
m.Set(key, value)
return true
}
return false
}
func (c *CMap) SetIf(key string, fn SetIfFunc) {
m := c.s.GetShard(key)
m.Lock()
defer m.Unlock()
v, ok := m.Get(key)
setValue, isSet := fn(ok, v)
if isSet {
m.Set(key, setValue)
}
}
func (c *CMap) RemoveIf(key string, fn RemoveIfFunc) (removed bool) {
m := c.s.GetShard(key)
m.Lock()
defer m.Unlock()
v, ok := m.Get(key)
remove := fn(ok, v)
if remove && ok {
m.Remove(key)
return true
}
return false
}