登录
首页 >  Golang >  Go问答

将值接收器添加到大小合适的切片

来源:stackoverflow

时间:2024-03-26 12:51:33 410浏览 收藏

在 Go 语言中,当将值接收器添加到大小合适的切片时,即使底层数组有足够容量,附加操作也无法正常工作。这是因为切片标头不会更改,而只有参数内的局部变量会更改。要对原始切片进行修改,需要返回一个新的切片,并在使用处重新赋值。

问题内容

有人可以帮助我理解这里发生了什么吗?

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 := make([]string, 0, 1)
fmt.println(cap(slice))
appendstring(slice, "a")
fmt.println(slice)

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

1
[]

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

1
[a]

事实并非如此,因为即使不会分配新的后备数组,main() 内的 slice 变量中的切片头不会更改,它仍然会保持 长度 = 0。仅更改了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]

好了,本文到此结束,带大家了解了《将值接收器添加到大小合适的切片》,希望本文对你有所帮助!关注golang学习网公众号,给大家分享更多Golang知识!

声明:本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>