卓越飞翔博客卓越飞翔博客

卓越飞翔 - 您值得收藏的技术分享站
技术文章34237本站已运行391

Go:附加第二项更改第一项

go:附加第二项更改第一项

问题内容

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

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}],即数组的第一个元素更改为插入的元素,尽管不应该如此。可能是什么原因以及如何解决它?

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


正确答案


虽然我同意其他人的评论,即修复方法是将 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在他们的示例中观察到的行为。由于切片是指向数组的指针,因此需要在 for 循环内声明 buf ,以便每次迭代附加不同的实例。

现在让我们看看如果我们对数组执行相同的操作会发生什么。

数组

引用来源的关键句子:

现在让我们看一个示例,该示例演示了在传递数组时(即,将其传递给 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 变量,但所有值都不同了。

卓越飞翔博客
上一篇: 在 Go 中映射 DTO 时减少重复代码的数量
下一篇: 返回列表
留言与评论(共有 0 条评论)
   
验证码:
隐藏边栏