如何在Go中处理切片的副本?

时间:2018-08-18 20:41:57

标签: go slice

mapArray是float32的2D切片。我对其进行了复制,因此我可以在不修改mapArray 的情况下进行复制。但是,事实并非如此。为from telethon import TelegramClient, types, sync with TelegramClient('name', api_id, api_hash) as client: message = client.get_messages('TelethonChat', ids=types.InputMessagePinned()) 分配值会修改Origin

mapArray

这样做会使origins := it.Empty2DArray(len(mapArray)) copy(origins, mapArray) origins[5][5] = -1 变为-1,而不是其原始值。

我如何制作切片的真实独立副本?

谢谢。

编辑:

mapArray[5][5]

1 个答案:

答案 0 :(得分:3)

2D切片是切片的切片。在您的函数中,您分配一个切片来容纳其他切片,然后为每个切片分配内存来容纳该数据行。要复制它,您需要复制所有这些数据行以及整个切片。

当您说copy(origins, mapArray)时,您真正要做的是复制指向原始数据的指针的一部分。但是,您不会复制原始数据。

我建议不要使用嵌套的for循环来复制子切片,而应使用一维切片并创建包装函数以对其进行索引。这样可以提高内存效率,并且可以使用内置的copy

这里是sample of what I would do instead

package main

import "fmt"

type squareMat struct {
    size int
    data []float32
}

func newSquareMat(size int) *squareMat {
    return &squareMat{
        size: size,
        data: make([]float32, size*size),
    }
}

func (s *squareMat) get(i, j int) float32 {
    return s.data[i+j*s.size]
}

func (s *squareMat) set(i, j int, to float32) {
    s.data[i+j*s.size] = to
}

func (s *squareMat) copy() *squareMat {
    c := newSquareMat(s.size)
    copy(c.data, s.data)
    return c
}

func main() {
    m := newSquareMat(5)
    m.set(2, 3, 1.5)
    n := m.copy()
    n.set(2, 3, 99)
    fmt.Println(m.get(2, 3))
    fmt.Println(n.get(2, 3))
}

如果您坚持使用2D float32数组,请here is how to copy that

package main

import "fmt"

func copy2D(x [][]float32) [][]float32 {
    c := make([][]float32, len(x))
    for i := range c {
        c[i] = make([]float32, len(x[i]))
        copy(c[i], x[i])
    }
    return c
}

func main() {
    a := [][]float32{
        []float32{1, 2, 3},
        []float32{4, 5, 6},
        []float32{7, 8, 9},
    }
    b := copy2D(a)
    b[1][1] = 99
    fmt.Println(a)
    fmt.Println(b)
}