我正在编写一个端点来返回Geckoboard的数据,它除了这样的格式:
{
"item": [
{
"value": "274057"
},
[
"38594",
"39957",
"35316",
"35913",
"36668",
"45660",
"41949"
]
]
}
"item"
是一组不同的结构。我如何在Go中表示这些数据?
注意:这不是关于我如何解组这个,我需要生成这种格式。
答案 0 :(得分:1)
这个东西比你想象的容易。对于休闲读者而言,这并没有很好的记录。我建议ffjson
超过普通json
。它的构成方式使您不需要更改库名称以外的语法。
这很容易:
type User struct {
Id int `json:'id'`
Name string `json:name`
SomeId1 int `json:some_id_1`
SomeId2 int `json:some_id_2`
SomeId3 int `json:some_id_3`
SomeId4 int `json:some_id_4`
}
item := map[string]User{}
for i := 0; i < 10; i++ {
item[strconv.itoa(i)] = User{i, "Username X", 38393, 29384, 12393, 123981}
}
buf, err := ffjson.Marshal(&item)
结构的缺点(即使在ffjson
仍然存在)是reflection
将始终使用,在你需要高性能的时候,你会浪费很多CPU周期。当你将它保存到地图时,ffjson
比普通json
快2-3倍。这样,库可以编译您编组的每个数据结构并重新使用它,而不是使用reflect
不断检查数据完整性/结构。
答案 1 :(得分:1)
有一种简单的方法可以创建一些数据结构的值,您可以在其中知道要生成/复制的JSON输出:
获取要生成的输出,并将其解组为map[string]interface{}
。您将获得一个地图值,当您编组时将产生您想要的输出。当您解组预期输出时,您可以检查结果映射值以了解您需要创建的内容以便输出预期的输出。
这也适用于您的情况。这是代码:
var m map[string]interface{}
err := json.Unmarshal([]byte(input), &m)
if err != nil {
panic(err)
}
fmt.Printf("%+v\n", m)
res, err := json.MarshalIndent(m, "", " ")
if err != nil {
panic(err)
}
fmt.Println(string(res))
{J}输入input
的位置:
const input = `{
"item": [
{
"value": "274057"
},
[
"38594",
"39957",
"35316",
"35913",
"36668",
"45660",
"41949"
]
]
}`
该程序生成与所需输出(或输入)相同的输出:
map[item:[map[value:274057] [38594 39957 35316 35913 36668 45660 41949]]]
{
"item": [
{
"value": "274057"
},
[
"38594",
"39957",
"35316",
"35913",
"36668",
"45660",
"41949"
]
]
}
在Go Playground上尝试完整的应用程序。
显然它有一个键"item"
,其值类型为:
fmt.Printf("%T\n", m["item"]); // Prints []interface{}
所以这是一个切片。它有两个值,它们的类型:
fmt.Printf("%T\n", m["item"].([]interface{})[0]); // map[string]interface{}
fmt.Printf("%T\n", m["item"].([]interface{})[1]); // []interface{}
因此切片包含2个值:
"value" : "274057"
对)以下是重现相同地图值的Go代码:
m := map[string]interface{}{
"item": []interface{}{
map[string]interface{}{"value": "274057"},
[]interface{}{"38594", "39957", "35316", "35913", "36668", "45660", "41949"},
},
}
Marshaling会产生相同的JSON输出。