// 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 }