我是初学者并尝试解组以下json字符串
[{
"db": {
"url": "mongodb://localhost",
"port": "27000",
"uname": "",
"pass": "",
"authdb": "",
"replicas": [
{
"rs01": {
"url":"mongodb://localhost",
"port": "27001",
"uname": "",
"pass": "",
"authdb": ""
}
},
{
"rs02": {
"url":"mongodb://localhost",
"port": "27002",
"uname": "",
"pass": "",
"authdb": ""
}
}
]
}
}]
这是结构
type DBS struct {
URL string `json:url`
Port string `json:port`
Uname string `json:uname`
Pass string `json:pass`
Authdb string `json:authdb`
Replicas []DBS `json:replicas`
}
这是函数
func loadConfigs() []DBS {
var config []DBS
raw, err := ioutil.ReadFile("./config.json")
if err != nil {
fmt.Println(err.Error())
os.Exit(1)
}
json.Unmarshal(raw, &config)
return config
}
该功能正在返回
{ []}
答案 0 :(得分:5)
您的JSON输入不是DBS
的片段,因为还有另一个JSON对象包装器,值DBS
属于属性"db"
。
更深入地说,"replicaps"
是一个JSON数组,对象包含不同的键,它们的值可由DBS
表示。
因此,要完整地描述您的JSON,您需要某种“动态”类型。例如,地图就是这样的动态类型。
因此,您的原始JSON输入可以使用类型[]map[string]DBS
进行完全建模。它是一片地图,因为你的JSON输入包含一个JSON数组。 map键可以为任何属性名称建模,值是由DBS
struct建模的JSON对象。
请参阅完整解析JSON输入的示例:
type DBS struct {
URL string `json:"url"`
Port string `json:"port"`
Uname string `json:"uname"`
Pass string `json:"pass"`
Authdb string `json:"authdb"`
Replicas []map[string]DBS `json:"replicas"`
}
func main() {
var dbs []map[string]DBS
if err := json.Unmarshal([]byte(src), &dbs); err != nil {
panic(err)
}
fmt.Printf("%+v", dbs)
}
请注意正确的tag syntax(例如json:"url"
)。
输出(在Go Playground上尝试):
[map[db:{URL:mongodb://localhost Port:27000 Uname: Pass: Authdb: Replicas:[map[rs01:{URL:mongodb://localhost Port:27001 Uname: Pass: Authdb: Replicas:[]}] map[rs02:{URL:mongodb://localhost Port:27002 Uname: Pass: Authdb: Replicas:[]}]]}]]
请注意,您可以进一步建模始终为"db"
的第一级,并且我们可以切换到指针(我在第一个示例中使用了非指针,因此打印结果是可读的):
type DBReplicated struct {
DB *DBS `json:"db"`
}
type DBS struct {
URL string `json:"url"`
Port string `json:"port"`
Uname string `json:"uname"`
Pass string `json:"pass"`
Authdb string `json:"authdb"`
Replicas []map[string]*DBS `json:"replicas"`
}
func main() {
var dbs []*DBReplicated
if err := json.Unmarshal([]byte(src), &dbs); err != nil {
panic(err)
}
db := dbs[0].DB
fmt.Printf("%+v\n", db)
for _, dbs := range db.Replicas {
for name, replica := range dbs {
fmt.Printf("%s: %+v\n", name, replica)
}
}
}
输出(在Go Playground上尝试):
&{URL:mongodb://localhost Port:27000 Uname: Pass: Authdb: Replicas:[map[rs01:0x10538200] map[rs02:0x10538240]]}
rs01: &{URL:mongodb://localhost Port:27001 Uname: Pass: Authdb: Replicas:[]}
rs02: &{URL:mongodb://localhost Port:27002 Uname: Pass: Authdb: Replicas:[]}
答案 1 :(得分:1)
如果出于某种原因,您仍然坚持使用现有的json.Unmarshaler
结构,那么您也可以编写自己的DBS
实现。
(在下面的示例中,我实际上会稍微更改结构以跟踪名称/键,但这是可选的,没有必要进行解组工作。)
type DBS struct {
name string
URL string `json:url`
Port string `json:port`
Uname string `json:uname`
Pass string `json:pass`
Authdb string `json:authdb`
Replicas []DBS `json:replicas`
}
func (db *DBS) UnmarshalJSON(data []byte) error {
raw := map[string]json.RawMessage{}
if err := json.Unmarshal(data, &raw); err != nil {
return err
}
if len(raw) > 1 {
return fmt.Errorf("fail")
}
type _DBS DBS
_db := (*_DBS)(db)
for name, v := range raw {
db.name = name
return json.Unmarshal(v, _db)
}
return nil
}