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:
切片是一个引用类型,那么就暂且用其他语言的引用类型去类比理解吧。