4

我认为在 GO 语言中,切片是通过引用传递的。但是为什么下面的代码没有改变 slice c 的内容呢?我错过了什么吗?谢谢你。

package main

import (
    "fmt"
)


func call(c []int) {
    c = append(c, 1)
    fmt.Println(c)
}

func main() {
    c := make([]int, 1, 5)
    fmt.Println(c)
    call(c)
    fmt.Println(c)
}

打印的结果是:

[0] [0 1] [0]

在我期待的时候

[0] [0 1] [0 1]

4

2 回答 2

12

切片的长度保存在不通过引用传递的切片头中。您可以将切片视为包含指向数组的指针、长度和容量的结构。

当您附加到切片时,您修改了数据数组中的索引 1,然后增加了切片标头中的长度。当您返回时,c主函数中的长度为 1,因此打印了相同的数据。

切片以这种方式工作的原因是您可以让多个切片指向相同的数据。例如:

x := []int{1,2,3}
y := x[:2] // [1 2]
z := x[1:] // [2 3]

所有这三个切片都指向同一底层数组中的重叠数据。

于 2012-09-05T03:44:55.743 回答
1

Go 总是按值传递。某些类型是引用类型,如指针、映射、通道;或部分引用类型,如切片(包括对底层数组的引用以及长度和容量的值)。但无论类型如何,一切都是按值传递的。因此,分配给局部变量永远不会影响外部的任何东西。

于 2012-09-05T19:26:39.367 回答