如何在golang中复制地图?

xak*_*p35 4 syntax dictionary go

我可以提供将地图分解为 2 个切片的示例:

func decomposeMap(m map[string]int) ([]string, []int) {
  var i uint
  l := len(m)
  keys, values := make([]string, l), make([]int, l)
  for keys[i], values[i] = range m {
    i++
  }
  return keys, values
}
Run Code Online (Sandbox Code Playgroud)

但我无法编写地图复制:

func copyMap(m map[string]int) map[string]int {
  m2 := make(map[string]int, len(m))
  for id, m2[id] = range m {} // error - id is not declared
  for id, m2[id] := range m {} // error with m2[id] already declared
  // id should not be accessible here, it should exist only inside loop
  return m2
}
Run Code Online (Sandbox Code Playgroud)

我可以将 id 声明为 var,但我不希望它在 for 循环之外可用。我如何混合分配和声明,例如:for id:=, m[id]= range m {}?那么它会在 for 循环内部声明索引,并且在外部无法访问吗?

icz*_*cza 6

id变量必须在 之前声明for,因为不能在 中使用短变量声明m2[id]

func copyMap(m map[string]int) map[string]int {
    m2 := make(map[string]int, len(m))
    var id string
    for id, m2[id] = range m {
    }
    return m2
}
Run Code Online (Sandbox Code Playgroud)

但!这不会重复地图!键仅在已评估id后才分配给m2[id],因此此循环会将值分配给前一次迭代的键,这不是重复,这是“洗牌”!

这基本上是一个元组分配(键和值分配给id, m2[id])。规格: 作业:

任务分两个阶段进行。首先,左边的索引表达式指针间接(包括选择器中的隐式指针间接)的操作数和右边的表达式都按通常的顺序进行计算。其次,作业按从左到右的顺序进行。

因此首先id对 和m2[id]进行求值(包括索引表达式),因此id“尚未”更改,因此使用上一次迭代的值,并且只有在此之后才分配新的键和值。

要演示,请参阅:

m := map[string]int{
    "one":   1,
    "two":   2,
    "three": 3,
}
m2 := copyMap(m)
fmt.Println(m2)
Run Code Online (Sandbox Code Playgroud)

输出(在Go Playground上尝试):

map[:1 one:2 two:3]
Run Code Online (Sandbox Code Playgroud)

这些值被分配给不同的键(与源映射中的不同),并且在第一次迭代中将一个值分配给空字符串键(默认值为零id)。

要复制地图,只需使用:

for id, v := range m {
    m2[id] = v
}
Run Code Online (Sandbox Code Playgroud)

或者,如果您想避免临时分配:

for id := range m {
    m2[id] = m[id]
}
Run Code Online (Sandbox Code Playgroud)