Go:附加第二个项目会更改第一个项目

问题描述 投票:0回答:1

不知道该说什么,只看代码和输出。 代码:

package main

import (
    "encoding/json"
    "fmt"
)

type Human struct {
    N string `json:"n"`
    C string `json:"c"`
}

func funcname(jDL ...[]byte) {
    jDL_len := len(jDL)
    people := [][]Human{}
    var buf []Human
    var err error
    for i := 0; i < jDL_len; i++ {
        err = json.Unmarshal(jDL[i], &buf)
        fmt.Println("buf", i, buf)
        people = append(people, buf)
        fmt.Println("people", i, people)
    }
    jsoned, err := json.Marshal(people)
    fmt.Println("jsoned", string(jsoned))
    fmt.Println("err", err)
}

func main() {
    i0 := []byte(`[
            {
                "n": "Alice",
                "c": "A1"
            },
            {
                "n": "Bob",
                "c": "A2"
            }
        ]`)
    i1 := []byte(`[
            {
                "n": "Clark",
                "c": "B1"
            },
            {
                "n": "Davis",
                "c": "B2"
            }
        ]`)
    funcname(i0, i1)
}

输出:

buf 0 [{Alice A1} {Bob A2}]
people 0 [[{Alice A1} {Bob A2}]]
buf 1 [{Clark B1} {Davis B2}]
people 1 [[{Clark B1} {Davis B2}] [{Clark B1} {Davis B2}]]
jsoned [[{"n":"Clark","c":"B1"},{"n":"Davis","c":"B2"}],[{"n":"Clark","c":"B1"},{"n":"Davis","c":"B2"}]]
err <nil>

查看输出中的 people 0 和 people 1 字符串。我使用 append 来合并 buf 0 和 buf 1,因此输出中的 people 1 字符串应如下所示:

[[{Alice A1} {Bob A2}]] [{Clark B1} {Davis B2}]
,即数组的第一个元素更改为插入的元素,尽管不应该如此。可能是什么原因以及如何解决?

我没有尝试任何修复,因为我在代码中没有发现任何问题。

arrays json go append
1个回答
0
投票

虽然我同意其他人的评论,即修复方法是将

buf
变量移动到 for 循环内,但我也相信这种混乱源于
buf
是一个切片,也许 OP 不知道切片是如何工作的。切片与数组不同,如Go 切片:用法和内部结构中所述。

切片

引用来源的关键句子:

切片是数组段的描述符。它由指向数组的指针、段的长度及其容量组成。

考虑到这一点,请看以下简化示例:

package main

import "fmt"

func main() {
    slice := [][]int{}
    buf := []int{0} // Slice

    for i := 0; i < 3; i++ {
        // Modifying the slice which is a pointer to an underlying array.
        buf[0] = i
        // Appending the slice (i.e. pointer).
        slice = append(slice, buf)
    }

    fmt.Println(slice)
}

输出:

[[2] [2] [2]]

这是OP在他们的示例中观察到的行为。现在让我们看看如果我们对数组执行相同的操作会发生什么。

数组

引用来源的关键句子:

Go 的数组是值。数组变量表示整个数组;它不是指向第一个数组元素的指针(如 C 中的情况)。这意味着当您分配或传递数组值时,您将复制其内容。

现在让我们看一个示例,该示例演示了在传递数组时(即,将其传递给

append
时)由数组内容组成的副本:

package main

import "fmt"

func main() {
    slice := [][1]int{}
    buf := [1]int{} // Array

    for i := 0; i < 3; i++ {
        // Modifying the array.
        buf[0] = i
        // Appending a copy of the array.
        slice = append(slice, buf)
    }

    fmt.Println(slice)
}

输出:

[[0] [1] [2]]

现在,尽管 for 循环中使用了相同的

buf
变量,但所有值都不同了。

© www.soinside.com 2019 - 2024. All rights reserved.