首页  >  文章  >  后端开发  >  为什么 Golang 的 `append()` 似乎在通过通道发送切片后修改它?

为什么 Golang 的 `append()` 似乎在通过通道发送切片后修改它?

Linda Hamilton
Linda Hamilton原创
2024-11-02 10:01:31263浏览

Why Does Golang's `append()` Seem to Modify a Slice After It's Sent Over a Channel?

Golang的append()什么时候创建一个新的切片?

append()的文档表明它会分配一个新的切片并且当原切片容量不足时,进行元素复制。但是,在检查以下代码的输出时会出现差异,该代码会生成布尔字母表的组合。

<code class="go">package main

import (
    "fmt"
)

func AddOption(c chan []bool, combo []bool, length int) {
    if length == 0 {
        fmt.Println(combo, "!")
        c <- combo
        return
    }
    var newCombo []bool
    for _, ch := range []bool{true, false} {
        newCombo = append(combo, ch)
        AddOption(c, newCombo, length-1)
    }
}

func main() {
    c := make(chan []bool)
    go func(c chan []bool) {
        defer close(c)
        AddOption(c, []bool{}, 4)
    }(c)
    for combination := range c {
        fmt.Println(combination)
    }
}</code>

对比观察

在输出中,以感叹号结尾的行表示 AddOption 通过通道发送的切片,而没有感叹号的行表示 main() 中接收的切片。值得注意的是,尽管append()据称返回了一个新切片,但通过通道发送的切片似乎在发送后被修改。

检查源

有问题的代码块是:

<code class="go">var newCombo []bool
for _, ch := range []bool{true, false} {
    newCombo = append(combo, ch)
    AddOption(c, newCombo, length-1)
}</code>

根据文档,当原始切片的容量不足时,append() 应该返回一个新的切片描述符,指向新的底层数据数组。但是,作为 AddOption 的第二个参数传递的值可以是指向切片描述符的指针,也可以是切片描述符的真实副本。

澄清行为

这个问题的答案在于区分切片数据类型和它的实际表示。切片描述符由长度和容量的两个整数以及指向底层数据的指针组成。

虽然append()确实返回一个带有可能不同的底层数据数组的新切片描述符,但指向数据的指针除非容量扩大,否则保持不变。这意味着虽然切片描述符本身是副本,但指针值(底层数据的地址)是共享的。

其他示例

为了说明,请考虑这个代码片段:

<code class="go">package main

import "fmt"

func main() {
    s := make([]int, 0, 5)
    s = append(s, []int{1, 2, 3, 4}...)

    a := append(s, 5)
    fmt.Println(a)

    b := append(s, 6)
    fmt.Println(b)
    fmt.Println(a)
}</code>

此代码打印:

[1 2 3 4 5]
[1 2 3 4 6]
[1 2 3 4 6]

最初,s有足够的容量,因此a和b共享相同的底层数据指针。但是,如果我们将 s 的容量减少到 4,输出将更改为:

[1 2 3 4 5]
[1 2 3 4 6]
[1 2 3 4 5]

这表明只有当 s 有足够的容量时,a 和 b 才会共享相同的底层数据。

以上是为什么 Golang 的 `append()` 似乎在通过通道发送切片后修改它?的详细内容。更多信息请关注PHP中文网其他相关文章!

声明:
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn