golang是不能傳引用的,所有的函數都是傳值。
遇到過坑,總結一下:
傳切片作爲形參:
func main() {
res := make([][]int, 0)
t := make([]int, 0)
TestCopy(&res, t)
fmt.Println("最後結果:", res)
}
func TestCopy(res *[][]int, t []int) {
for i := 0; i < 5; i++ {
t = append(t, i)
*res = append(*res, t)
fmt.Println("在for循環中: ", t, *res)
}
t[0] = -1
fmt.Println("改變了t[0]的值後: ", t, *res)
}
輸出結果:
在for循環中: [0] [[0]]
在for循環中: [0 1] [[0] [0 1]]
在for循環中: [0 1 2] [[0] [0 1] [0 1 2]]
在for循環中: [0 1 2 3] [[0] [0 1] [0 1 2] [0 1 2 3]]
在for循環中: [0 1 2 3 4] [[0] [0 1] [0 1 2] [0 1 2 3] [0 1 2 3 4]]
改變了t[0]的值後: [-1 1 2 3 4] [[0] [0 1] [0 1 2] [0 1 2 3] [-1 1 2 3 4]]
最後結果: [[0] [0 1] [0 1 2] [0 1 2 3] [-1 1 2 3 4]]
根據go只傳值的特性和切片本身就是一個引用的特點,就算在for循環之外對t[0]進行修改,都會影響到t甚至是res的值。
那麼可以想見,在進行 *res = append(*res, t) 操作時,就是將t和後面的一串地址傳到了res中。
所以,在傳一個切片,或者把切片放入其他位置時,一定要注意再次修改對原值的影響。
此時,可以使用copy()來完成深拷貝。
ps:
切片是一個引用類型,那麼就暫且用其他語言的引用類型去類比理解吧。