将字符串作为参数传递时要复制什么?

What is being copied when passing a string as parameter?

在Golang中,一切都是按值传递的。如果我“直接”传递一个数组(而不是通过指针传递它),那么在函数中所做的任何修改都会在它之外找到

func f(a []int) {
    a[0] = 10
}
func main() {
    a := []int{2,3,4}
    f(a)
    fmt.Println(a)
}

Output: [10 3 4]

这是因为,根据我的理解,数组构成(除其他外)指向基础数据数组的指针。

除非我弄错了(参见 here),字符串还构成(与“len”对象一起)指向基础数据的指针(unsafe.Pointer)。因此,我期待与上述相同的行为,但显然我错了。

func f(s string) {
    s = "bar"
}
func main() {
    s := "foo"
    f(s)
    fmt.Println(s)
}

Output: "foo"

这里的字符串是怎么回事?当字符串作为参数传递时,似乎正在复制基础数据。

相关问题:当​​我们不希望我们的函数修改字符串时,出于性能原因是否仍然建议通过指针传递大字符串?

A string 中有两个值:指向数组的指针和字符串长度。当您将字符串作为参数传递时,将复制这两个值,而不是基础数组。

除了使用 unsafe 之外,没有办法修改字符串的内容。当您将 *string 传递给函数并且该函数修改字符串时,该函数只是修改字符串以指向不同的数组。