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

Pointers 使用值接收器附加到具有足够容量的片,pointers,go,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) } 我

有人能帮我理解这里发生了什么吗

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)
}
我知道片头和使用指针接收器的必要性。但是在这里,由于基础数组有足够的容量,我希望append无论如何都能正常工作(只需将新值添加到基础数组中,原始的[copied]头就可以正常工作)


我的假设有什么问题?

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

slice := make([]string, 0, 1)
fmt.Println(cap(slice))
appendString(slice, "a")
fmt.Println(slice)
输出将是(在上尝试):

这是正确的。人们可以预期结果是:

1
[a]
情况并非如此的原因是,即使不会分配新的后备数组,
main()
slice
变量中的slice头未更改,它仍将保持
length=0
。只有存储在
slice
局部变量
appendString()
(参数)中的slice头被更改,但该变量独立于main的
slice

如果要重新切片main的
切片
,您将看到备份数组确实包含新字符串:

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

slice = slice[:1]
fmt.Println(slice)
现在输出将是(在上尝试):

这就是内置必须返回新切片的原因:因为即使不需要新的后备数组,如果附加了超过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)
然后你马上就得到了预期的结果(在网上试试):


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

slice := make([]string, 0, 1)
fmt.Println(cap(slice))
appendString(slice, "a")
fmt.Println(slice)
输出将是(在上尝试):

这是正确的。人们可以预期结果是:

1
[a]
情况并非如此的原因是,即使不会分配新的后备数组,
main()
slice
变量中的slice头未更改,它仍将保持
length=0
。只有存储在
slice
局部变量
appendString()
(参数)中的slice头被更改,但该变量独立于main的
slice

如果要重新切片main的
切片
,您将看到备份数组确实包含新字符串:

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

slice = slice[:1]
fmt.Println(slice)
现在输出将是(在上尝试):

这就是内置必须返回新切片的原因:因为即使不需要新的后备数组,如果附加了超过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)
然后你马上就得到了预期的结果(在网上试试):


append
始终返回新的切片标头,因为长度字段必须更改。您所考虑的只是底层数组,它不需要在这里重新分配。但是切片总是会更改。@Peter不总是,只有当追加了0个以上的元素时才是如此。@Peter这正是我的问题
append
总是返回一个新的切片标题,因为长度字段必须更改。您所考虑的只是底层数组,它不需要在这里重新分配。但是切片总是变化的。@Peter不总是,只有在附加了超过0个元素的情况下才是如此。@Peter这正是我的问题伟大的答案,我只是考虑了支持数组,而不是长度值。谢谢。回答得很好,我只是在考虑支持数组,而不是长度值。谢谢