mirror of
https://github.com/AdguardTeam/AdGuardHome.git
synced 2024-11-24 14:05:45 +03:00
87 lines
1.7 KiB
Go
87 lines
1.7 KiB
Go
|
package aghalg
|
||
|
|
||
|
import (
|
||
|
"slices"
|
||
|
)
|
||
|
|
||
|
// SortedMap is a map that keeps elements in order with internal sorting
|
||
|
// function. Must be initialised by the [NewSortedMap].
|
||
|
type SortedMap[K comparable, V any] struct {
|
||
|
vals map[K]V
|
||
|
cmp func(a, b K) (res int)
|
||
|
keys []K
|
||
|
}
|
||
|
|
||
|
// NewSortedMap initializes the new instance of sorted map. cmp is a sort
|
||
|
// function to keep elements in order.
|
||
|
//
|
||
|
// TODO(s.chzhen): Use cmp.Compare in Go 1.21.
|
||
|
func NewSortedMap[K comparable, V any](cmp func(a, b K) (res int)) SortedMap[K, V] {
|
||
|
return SortedMap[K, V]{
|
||
|
vals: map[K]V{},
|
||
|
cmp: cmp,
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Set adds val with key to the sorted map. It panics if the m is nil.
|
||
|
func (m *SortedMap[K, V]) Set(key K, val V) {
|
||
|
m.vals[key] = val
|
||
|
|
||
|
i, has := slices.BinarySearchFunc(m.keys, key, m.cmp)
|
||
|
if has {
|
||
|
m.keys[i] = key
|
||
|
} else {
|
||
|
m.keys = slices.Insert(m.keys, i, key)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Get returns val by key from the sorted map.
|
||
|
func (m *SortedMap[K, V]) Get(key K) (val V, ok bool) {
|
||
|
if m == nil {
|
||
|
return
|
||
|
}
|
||
|
|
||
|
val, ok = m.vals[key]
|
||
|
|
||
|
return val, ok
|
||
|
}
|
||
|
|
||
|
// Del removes the value by key from the sorted map.
|
||
|
func (m *SortedMap[K, V]) Del(key K) {
|
||
|
if m == nil {
|
||
|
return
|
||
|
}
|
||
|
|
||
|
if _, has := m.vals[key]; !has {
|
||
|
return
|
||
|
}
|
||
|
|
||
|
delete(m.vals, key)
|
||
|
i, _ := slices.BinarySearchFunc(m.keys, key, m.cmp)
|
||
|
m.keys = slices.Delete(m.keys, i, i+1)
|
||
|
}
|
||
|
|
||
|
// Clear removes all elements from the sorted map.
|
||
|
func (m *SortedMap[K, V]) Clear() {
|
||
|
if m == nil {
|
||
|
return
|
||
|
}
|
||
|
|
||
|
m.keys = nil
|
||
|
clear(m.vals)
|
||
|
}
|
||
|
|
||
|
// Range calls cb for each element of the map, sorted by m.cmp. If cb returns
|
||
|
// false it stops.
|
||
|
func (m *SortedMap[K, V]) Range(cb func(K, V) (cont bool)) {
|
||
|
if m == nil {
|
||
|
return
|
||
|
}
|
||
|
|
||
|
for _, k := range m.keys {
|
||
|
if !cb(k, m.vals[k]) {
|
||
|
return
|
||
|
}
|
||
|
}
|
||
|
}
|