使用值接收器附加到具有足够容量的切片

时间:2018-10-26 13:10:41

标签: pointers go append slice

有人可以帮助我了解这里发生的事情吗?

package main

import (
    "fmt"
)

func appendString(slice []string, newString string) {
    slice = append(slice, newString)
}

func main() {
    slice := make([]string, 0, 1)
    appendString(slice, "a")
    fmt.Println(slice)
}

我了解切片标头以及使用指针接收器的需要。但是在这里,由于基础数组具有足够的容量,所以我希望追加仍可以正常工作(只需将新值添加到基础数组中,并且原始的[copied]标头可以按预期工作)

我的假设有什么问题?

1 个答案:

答案 0 :(得分:4)

让我们添加最终的打印语句以查看结果:

slice := make([]string, 0, 1)
fmt.Println(cap(slice))
appendString(slice, "a")
fmt.Println(slice)

输出将是(在Go Playground上尝试):

1
[]

这是正确的。可以期望输出为:

1
[a]

之所以不是这样,是因为即使不分配新的后备数组,在slice内的main()变量中的slice头也不会被更改。 >,它将仍然保留length = 0。只有存储在slice(参数)内的appendString()局部变量中的slice头会更改,但是此变量与main的slice是独立的。

如果要重新分配main的slice,您会看到支持数组确实包含新字符串:

slice := make([]string, 0, 1)
fmt.Println(cap(slice))
appendString(slice, "a")
fmt.Println(slice)

slice = slice[:1]
fmt.Println(slice)

现在将输出(在Go Playground上尝试):

1
[]
[a]

这就是内置append()必须返回新切片的原因:因为即使不需要新的后备数组,切片标头(包含长度)也必须更改(增加),如果大于附加了0个元素。

这就是appendString()还应该返回新切片的原因:

func appendString(slice []string, newString string) []string {
    slice = append(slice, newString)
    return slice
}

或更短:

func appendString(slice []string, newString string) []string {
    return append(slice, newString)
}

您必须在使用它的地方重新分配:

slice := make([]string, 0, 1)
fmt.Println(cap(slice))
slice = appendString(slice, "a")
fmt.Println(slice)

然后立即获得预期的结果(在Go Playground上尝试):

1
[a]