Go 语言中 append() 函数:为什么对切片 x 的修改会影响 y 和 z?

go 语言中 append() 函数:为什么对切片 x 的修改会影响 y 和 z?

append() 方法的困惑

go 语言中的 append() 函数用于在切片的末尾添加元素。然而,在某些情况下,它可能会产生意想不到的结果。

考虑以下代码:

package main

import "fmt"

func main()  {
    x := make([]int,0,10)
    x = append(x, 1,2,3)
    y := append(x,4)
    z := append(x,5)
    fmt.Println(x)
    fmt.Println(y)
    fmt.Println(z)
}

问题在于,在对 x 执行 append(x, 5) 后,y 中的元素也发生了改变,即使 append() 通常被认为只复制 x 的值。

理解切片类型

要理解这种情况,就需要对 go 中的切片类型有一个明确的认识。切片是一个包含底层数组指针和长度的结构值。重要的是要记住,切片是一个值,而不是一个引用。

append() 的行为

append 被调用时,它会执行以下操作:

  • 如果目标切片的容量足够,它将直接在底层数组上修改值并返回一个更新的切片。
  • 如果目标切片的容量不足,它将重新分配一个新的底层数组并返回一个更新的切片。

代码分析

在给定的代码中:

  • 第一次 append (即 append(x, 1,2,3) ) 将元素添加到 x 中,并更新其长度。
  • 第二次 append (即 append(x,4) ) 创建了一个新的切片 y,该切片指向与 x 相同的底层数组,但长度增加了 1。
  • 第三次 append (即 append(x,5) ) 由于 x 的容量足够,因此直接修改了底层数组并更新了 x。

由于 y 和 z 指向与 x 相同的底层数组,因此对 x 所做的修改也会反映在 y 和 z 中,即使 y 和 z 具有不同的长度。

结论

为了避免此类问题,务必注意切片的复制语义。在需要使用副本时,可以手动复制切片,也可以使用 copy() 内置函数。

以上就是Go 语言中 append() 函数:为什么对切片 x 的修改会影响 y 和 z?的详细内容,更多请关注其它相关文章!