如何在Go中将JSON解组到接口{}?

时间:2015-01-31 16:56:28

标签: json go

我是Go的新手,现在我遇到了问题。我有一个名为Message的类型,它是这样的结构:

type Message struct {
    Cmd string `json:"cmd"`
    Data interface{} `json:"data"`
}

我也有一个名为CreateMessage的类型:

type CreateMessage struct {
    Conf map[string]int `json:"conf"`
    Info map[string]int `json:"info"`
}

我有一个像{"cmd":"create","data":{"conf":{"a":1},"info":{"b":2}}}这样的JSON字符串。

当我使用json.Unmarshal将其解码为Message变量时,答案为{Cmd:create Data:map[conf:map[a:1] info:map[b:2]]}

那么我可以将JSON解码为Message结构并将其Data的接口{}更改为根据Cmd键入CreateMessage吗?

我尝试将数据直接转换为CreateMessage类型,但编译器告诉我Data是一个map [sting] interface {}类型。

1 个答案:

答案 0 :(得分:34)

使用json.RawMessage字段为消息的固定部分定义结构类型,以捕获消息的变体部分。为每个变体类型定义结构类型,并根据命令对它们进行解码。

type Message struct {
  Cmd string `json:"cmd"`
  Data      json.RawMessage
}  

type CreateMessage struct {
    Conf map[string]int `json:"conf"`
    Info map[string]int `json:"info"`
}  

func main() {
    var m Message
    if err := json.Unmarshal(data, &m); err != nil {
        log.Fatal(err)
    }
    switch m.Cmd {
    case "create":
        var cm CreateMessage
        if err := json.Unmarshal([]byte(m.Data), &cm); err != nil {
            log.Fatal(err)
        }
        fmt.Println(m.Cmd, cm.Conf, cm.Info)
    default:
        log.Fatal("bad command")
    }
}

playground example