pointers - 使用值接收器 append 到具有足够容量的 slice

标签 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)
}

我知道 slice 头和使用指针接收器的需要。但在这里,由于底层数组有足够的容量,我希望 append 无论如何都能工作(只需将新值添加到底层数组,原始 [copied] header 按预期工作)

我的假设有什么问题?

最佳答案

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

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

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

1
[]

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

1
[a]

之所以不是这种情况,是因为即使不会分配新的支持数组,main() 中的 slice 变量中的 slice header 也是未更改,它仍将保持 length = 0。只有appendString()(参数)内部的slice局部变量中存储的slice header被改变了,但是这个变量独立于main的slice .

如果您要重新 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()必须返回新的 slice :因为即使不需要新的支持数组,如果 append 了超过 0 个元素, slice 头(包含长度)也必须更改(增加)。

这就是为什么 appendString() 也应该返回新的 slice :

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]

关于pointers - 使用值接收器 append 到具有足够容量的 slice ,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/53009514/

相关文章:

google-app-engine - 为 go gae 共享一个本地主机?

javascript - 有没有办法使用 Javascript 在 DIV 中找到最后一个元素的 ID?

c - 我如何理解复杂的函数声明?

c++ - 在将 void* 转换为任何内容时,我应该使用 static_cast 还是 reinterpret_cast

go - 无法通过 var.MethodName() 从另一个包访问方法

linux - 使用 sed - shell 脚本将值 append 到变量

javascript - 如何将我用 js append 的内容居中并替换它

c - 如何在 C 中打印出现的字母和单词(鼓励使用指针)

c - 获取数组元素的地址

go - 在 Go 中设置引用