了解 Go 中的切片可变性
在 Go 中,切片是一个动态大小的变量,用于存储连续的元素序列。它是使用存储长度和指向包含元素的底层数组的指针来实现的。
考虑以下代码片段:
func someFunc(A []int) int { ... tempA := A // copy the slice by value ... newArr = remove(tempA, i) ... } func remove(slice []int, s int) []int { return append(slice[:s], slice[s+1:]...) }
在此代码中,append 函数用于修改tempA切片。然而,当 someFunc 函数返回时,原始 A 切片也会被修改,尽管 tempA 是作为 A 的副本创建的。
为什么会发生这种情况?
切片头是与底层数组分开的数据结构。复制切片时,仅复制标头,而不复制底层数组。这意味着原始 A 切片和新创建的 tempA 切片都指向同一个底层数组。
调用append 函数时,会创建一个新切片,并将 tempA 切片的元素复制到新切片。然而,A 切片仍然指向相同的底层数组,该数组现在包含修改后的元素。
建议
为了避免在修改 a 时原始切片发生意外突变复制时,您应该使用复制功能来创建切片的深层复制。复制函数创建一个新的底层数组并将元素复制到其中。
以下是如何使用复制函数的示例:
tempA := make([]int, len(A)) copy(tempA, A)
这将为以下对象创建一个新的底层数组: tempA 并将 A 中的元素复制到其中。对 tempA 所做的任何修改都不会影响原始 A 切片。
以上是为什么复制 Go 切片不会创建深层复制,如何避免意外突变?的详细内容。更多信息请关注PHP中文网其他相关文章!