切片删除元素后原切片为何会改变?
在 go 中,切片是一种动态数组,其通过指向底层数组实现。当切片通过 append 函数修改时,可能会出现切片原值发生改变的情况。
考虑以下代码:
var i = 3 nums := []int{1, 2, 3, 4, 5, 6} nums6 := append(nums[:i], nums[i+2:]...) fmt.println(nums, nums6)
输出结果为:
// [1 2 3 6 5 6] [1 2 3 6]
预期中,nums 切片应变为 [1, 2, 3, 6],但实际上却保留了 5 和 6。这是因为:
-
append 函数会在以下两种情况下保留底层数组:
- 追加的数据不会超过底层数组的容量。
- 追加后,底层数组仍有剩余空间。
在给定的示例中,nums 切片的容量为 6,而删除的元素数量为 2。因此,append 函数会保留原底层数组,并直接修改其数据。当 nums6 新切片创建时,它将指向与 nums 相同的底层数组,导致 nums 切片也发生了改变。
反之,如果删除的元素数量导致底层数组的空间不足,append 函数会重新分配一个容量更大的底层数组,而 nums 切片将指向新分配的数组,从而不会发生改变。