登录
首页 >  Golang >  Go问答

GO呈现出复杂的层级关系

来源:stackoverflow

时间:2024-02-19 17:57:25 456浏览 收藏

Golang小白一枚,正在不断学习积累知识,现将学习到的知识记录一下,也是将我的所得分享给大家!而今天这篇文章《GO呈现出复杂的层级关系》带大家来了解一下##content_title##,希望对大家的知识积累有所帮助,从而弥补自己的不足,助力实战开发!


问题内容

我想澄清如何设置值

type elkbulkinsert struct {
    index []struct {
        _index string `json:"_index"`
        _id    string `json:"_id"`
    } `json:"index"`
}

制作 json.marshall 通常的结构没有问题

package main
import (
    "encoding/json"
    "fmt"
)
type ElkBulkInsert struct {
    Index []struct {
        _Index string `json:"_index"`
        _Id    string `json:"_id"`
    } `json:"index"`
}
type ElkBulIsertUrl struct {
    Url string `json:"url"`
}
func main() {
    sf := ElkBulIsertUrl{
        Url: "http://mail.ru",
    }
    dd, _ := json.Marshal(sf)
    fmt.Println(string(dd))
}

正确答案


真的不清楚你在这里问什么。您是否在问为什么 json 输出与您的期望不符?您不确定如何初始化/设置 []struct{...} 类型的 index 字段的值吗?

因为还不清楚,所以我将尝试解释为什么您的 json 输出可能会缺少字段(或者为什么不是所有字段都被填充)、如何初始化字段以及如何改进您拥有的类型。

一般答案

如果您想编组/解组到您创建的结构/类型中,请记住一个简单的规则:

json.marshaljson.unmarshal 只能访问导出字段。导出的字段具有大写标识符。 elkbulkinsert 中的 index 字段是匿名结构的一部分,该结构没有导出字段(_index_id 以下划线开头)。
因为无论如何您都使用 json:"_index" 标签,所以字段名称本身不必与 json 本身的字段类似。在大多数情况下,这显然是更好的做法,但这不是必需的。顺便说一句:您有一个名为 url 的字段。通常认为遵循标准 WRT initialisms 并将该字段重命名为 url 是更好的形式:

名称中的首字母缩写词或首字母缩略词(例如“url”或“nato”)具有一致的大小写。例如,“url”应显示为“url”或“url”(如“urlpony”或“urlpony”),而不是“url”。下面是一个示例:servehttp 而不是 servehttp。

当“id”是“identifier”的缩写时,此规则也适用,因此请写“appid”而不是“appid”。

协议缓冲区编译器生成的代码不受此规则的约束。人类编写的代码比机器编写的代码具有更高的标准。

话虽这么说,只需将类型更改为此即可:

type elkbulkinsert struct {
    index []struct {
        index string `json:"_index"`
        id    string `json:"_id"`
    } `json:"index"`
}

type elkbuliserturl struct {
    url string `json:"url"`
}

当然,这意味着 elkbulkinsert 的数据类似于:

{
    "index": [
        {
            "_index": "foo",
            "_id": "bar"
        },
        {
            "_index": "fizz",
            "_id": "buzz"
        }
    ]
}

当您想要为这样的结构设置值时,我通常发现避免使用匿名结构字段(例如 index 切片中的字段)会更容易,并使用类似以下内容的内容:

type elkinsertindex struct {
    id    string `json:"_id"`
    index string `json:"_index"`
}

type elkbulkinsert struct {
    index []elkinsertindex `json:"index"`
}

这使得填充切片变得更加容易:

bulk := elkbulkinsert{
    index: make([]elkinsertindex, 0, 10), // as an example
}
for i := 0; i < 10; i++ {
    bulk.index = append(bulk.index, elkinsertindex{
        id:    fmt.sprintf("%d", i),
        index: fmt.sprintf("idx@%d", i), // wherever these values come from
    })
}

甚至更简单(例如,在编写装置或单元测试时)是创建一个预先填充的文字:

data := elkbulkinsert{
    index: []elkinsertindex{
        {
            id:    "foo",
            index: "bar",
        },
        {
            id:    "fizz",
            index: "buzz",
        },
    },
}

对于您当前的类型,使用匿名结构类型,您仍然可以做同样的事情,但它看起来更混乱,并且需要更多维护:您必须重复该类型:

data := ElkBulkInsert{
    Index: []struct{
        ID    string `json:"_id"`
        Index string `json:"_index"`
    } {
        ID:    "foo",
        Index: "bar",
    },
    { // you can omit the field names if you know the order, and initialise all of them
        "fizz",
        "buzz",
    },
}

在这两种情况下都可能在初始化时省略字段名称,但我建议不要这样做。随着时间的推移,字段被添加/重命名/移动,维护这种混乱变成了一场噩梦。这也是为什么我强烈建议您远离此处的匿名结构。它们在某些地方可能很有用,但是当您表示来自外部方或发送到外部方的已知数据格式时(如 json 往往所做的那样),我发现最好对所有相关类型进行命名、标记,记录在某处。至少,您可以向每种类型添加注释,详细说明它代表什么值,并且您可以从中生成一个漂亮的、内容丰富的 godoc

本篇关于《GO呈现出复杂的层级关系》的介绍就到此结束啦,但是学无止境,想要了解学习更多关于Golang的相关知识,请关注golang学习网公众号!

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