如何在不更改原始数组的情况下复制数组?

我想复制一个数组,但复制的数组也总是对原始数组进行更改。为什么会这样?

以下是该问题的代码示例:

package main

import (
    "fmt"
)

func main() {
    array := make([]int, 2)
    for i := range array {
        array[i] = 0
    }

    oldArray := array
    array[0] = 3 // why does this also change oldArray?
    oldArray[1] = 2 // why does this also change the original array?

    fmt.Println(oldArray, array)
    // I expected [0,2], [3,0]
    // but it returns [3,2], [3,2]

}

我之前尝试启动变量oldArrayarray但结果是一样的。

回答

在您的示例代码中,您使用的是slices,而不是arrays。

从切片文档:

切片是底层数组的连续段的描述符,并提供对来自该数组的元素的编号序列的访问。

当您将切片分配给变量时,您正在创建该描述符的副本,因此处理相同的底层数组。当您实际使用数组时,它具有您期望的行为。

切片文档中的另一个片段(重点是我的):

切片一旦初始化,就始终与保存其元素的底层数组相关联。因此,切片与其数组以及同一数组的其他切片共享存储;相比之下,不同的数组总是代表不同的存储

这是一个代码示例(对于切片,第一个元素的内存地址在括号中,以明确指出两个切片何时使用相同的底层数组):

package main

import (
    "fmt"
)

func main() {
    // Arrays
    var array [2]int
    newArray := array
    array[0] = 3
    newArray[1] = 2
    fmt.Printf("Arrays:narray: %vnnewArray: %vnn", array, newArray)

    // Slices (using copy())
    slice := make([]int, 2)
    newSlice := make([]int, len(slice))
    copy(newSlice, slice)
    slice[0] = 3
    newSlice[1] = 2
    fmt.Printf("Slices (different arrays):nslice (%p): %v nnewSlice (%p): %vnn", slice, slice, newSlice, newSlice)

    // Slices (same underlying array)
    slice2 := make([]int, 2)
    newSlice2 := slice2
    slice2[0] = 3
    newSlice2[1] = 2
    fmt.Printf("Slices (same array):nslice2 (%p): %v nnewSlice2 (%p): %vnn", slice2, slice2, newSlice2, newSlice2)
}

输出:

Arrays:
array: [3 0]
newArray: [0 2]

Slices (different arrays):
slice (0xc000100040): [3 0] 
newSlice (0xc000100050): [0 2]

Slices (same array):
slice2 (0xc000100080): [3 2] 
newSlice2 (0xc000100080): [3 2]

去游乐场


以上是如何在不更改原始数组的情况下复制数组?的全部内容。
THE END
分享
二维码
< <上一篇
下一篇>>