-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathslmmap.go
66 lines (54 loc) · 1.38 KB
/
slmmap.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
package genh
import (
"hash/maphash"
"runtime"
"sync"
)
// SLMultiMap is segmented multimap that uses NumCPU maps to store data
// which lowers the lock contention.
type SLMultiMap[V any] struct {
ms []*LMultiMap[string, string, V]
s maphash.Seed
o sync.Once
}
func (lm *SLMultiMap[V]) m(k string) *LMultiMap[string, string, V] {
lm.o.Do(func() {
if len(lm.ms) == 0 {
lm.init(runtime.NumCPU())
}
})
return lm.ms[maphash.String(lm.s, k)%uint64(len(lm.ms))]
}
func (lm *SLMultiMap[V]) init(sz int) {
lm.ms = make([]*LMultiMap[string, string, V], sz)
for i := range lm.ms {
lm.ms[i] = NewLMultiMap[string, string, V](0)
}
lm.s = maphash.MakeSeed()
}
func (lm *SLMultiMap[V]) Set(k1, k2 string, v V) {
lm.m(k1).Set(k1, k2, v)
}
func (lm *SLMultiMap[V]) Get(k1, k2 string) V {
return lm.m(k1).Get(k1, k2)
}
func (lm *SLMultiMap[V]) MustGet(k1, k2 string, fn func() V) V {
return lm.m(k1).MustGet(k1, k2, fn)
}
func (lm *SLMultiMap[V]) Delete(k1, k2 string) {
lm.m(k1).Delete(k2)
}
func (lm *SLMultiMap[V]) Clear() {
for _, m := range lm.ms {
m.Clear()
}
}
func (lm *SLMultiMap[V]) ForEachChild(k1 string, fn func(k2 string, v V) bool) {
lm.m(k1).ForEachChild(k1, fn)
}
func (lm *SLMultiMap[V]) Update(k1 string, fn func(m map[string]V) map[string]V) {
lm.m(k1).Update(k1, fn)
}
func (lm *SLMultiMap[V]) SetChild(k1 string, v map[string]V) {
lm.m(k1).SetChild(k1, v)
}