mirror of
https://github.com/duke-git/lancet.git
synced 2026-02-04 12:52:28 +08:00
107 lines
2.2 KiB
Go
107 lines
2.2 KiB
Go
// Copyright 2021 dudaodong@gmail.com. All rights reserved.
|
|
// Use of this source code is governed by MIT license
|
|
|
|
// Package maputil includes some functions to manipulate map.
|
|
package maputil
|
|
|
|
import "reflect"
|
|
|
|
// Keys returns a slice of the map's keys
|
|
func Keys[K comparable, V any](m map[K]V) []K {
|
|
keys := make([]K, 0, len(m))
|
|
|
|
for k := range m {
|
|
keys = append(keys, k)
|
|
}
|
|
|
|
return keys
|
|
}
|
|
|
|
// Values returns a slice of the map's values
|
|
func Values[K comparable, V any](m map[K]V) []V {
|
|
values := make([]V, 0, len(m))
|
|
|
|
for _, v := range m {
|
|
values = append(values, v)
|
|
}
|
|
|
|
return values
|
|
}
|
|
|
|
// Merge maps, next key will overwrite previous key
|
|
func Merge[K comparable, V any](maps ...map[K]V) map[K]V {
|
|
res := make(map[K]V, 0)
|
|
|
|
for _, m := range maps {
|
|
for k, v := range m {
|
|
res[k] = v
|
|
}
|
|
}
|
|
|
|
return res
|
|
}
|
|
|
|
// ForEach executes iteratee funcation for every key and value pair in map
|
|
func ForEach[K comparable, V any](m map[K]V, iteratee func(key K, value V)) {
|
|
for k, v := range m {
|
|
iteratee(k, v)
|
|
}
|
|
}
|
|
|
|
// Filter iterates over map, return a new map contains all key and value pairs pass the predicate function
|
|
func Filter[K comparable, V any](m map[K]V, predicate func(key K, value V) bool) map[K]V {
|
|
res := make(map[K]V)
|
|
|
|
for k, v := range m {
|
|
if predicate(k, v) {
|
|
res[k] = v
|
|
}
|
|
}
|
|
return res
|
|
}
|
|
|
|
// Intersect iterates over maps, return a new map of key and value pairs in all given maps
|
|
func Intersect[K comparable, V any](maps ...map[K]V) map[K]V {
|
|
if len(maps) == 0 {
|
|
return map[K]V{}
|
|
}
|
|
if len(maps) == 1 {
|
|
return maps[0]
|
|
}
|
|
|
|
var res map[K]V
|
|
|
|
reducer := func(m1, m2 map[K]V) map[K]V {
|
|
m := make(map[K]V)
|
|
for k, v1 := range m1 {
|
|
if v2, ok := m2[k]; ok && reflect.DeepEqual(v1, v2) {
|
|
m[k] = v1
|
|
}
|
|
}
|
|
return m
|
|
}
|
|
|
|
reduceMaps := make([]map[K]V, 2, 2)
|
|
res = reducer(maps[0], maps[1])
|
|
|
|
for i := 2; i < len(maps); i++ {
|
|
reduceMaps[0] = res
|
|
reduceMaps[1] = maps[i]
|
|
res = reducer(reduceMaps[0], reduceMaps[1])
|
|
}
|
|
|
|
return res
|
|
}
|
|
|
|
// Minus creates an map of whose key in mapA but not in mapB
|
|
func Minus[K comparable, V any](mapA, mapB map[K]V) map[K]V {
|
|
res := make(map[K]V)
|
|
|
|
for k, v := range mapA {
|
|
if _, ok := mapB[k]; !ok {
|
|
res[k] = v
|
|
}
|
|
}
|
|
return res
|
|
}
|