一尘不染

在Go中解组不一致的JSON

go

我正在使用JSON,它返回三种不同的对象类型“项目”,“类别”和“修饰符”。可以在此处查看JSON的示例。我为三种类型的对象创建了模型。但是当我解组时,我选择了一种类型来解组整个JSON。(我知道这不可能是正确的方法…)然后,我尝试根据不同的类型来解析不同的项目,如json字段“类型”,然后将该对象附加到适当类型的切片中。我遇到错误,因为我不知道如何解组其中具有不同字段且具有不同类型的JSON。

解封包含不同对象的JSON的正确方法是什么,每个对象都有各自的字段?

是创建包含所有可能字段然后解组的“超级模型”的解决方案吗?

我还很新,不胜感激。谢谢!


阅读 300

收藏
2020-07-02

共1个答案

一尘不染

如果实现json.Unmarshaler,则可以定义一个将每种项目类型解析为相关结构的结构。

例:

// Dynamic represents an item of any type.
type Dynamic struct {
    Value interface{}
}

// UnmarshalJSON is called by the json package when we ask it to
// parse something into Dynamic.
func (d *Dynamic) UnmarshalJSON(data []byte) error {
    // Parse only the "type" field first.
    var meta struct {
        Type string
    }
    if err := json.Unmarshal(data, &meta); err != nil {
        return err
    }

    // Determine which struct to unmarshal into according to "type".
    switch meta.Type {
    case "product":
        d.Value = &Product{}
    case "post":
        d.Value = &Post{}
    default:
        return fmt.Errorf("%q is an invalid item type", meta.Type)
    }

    return json.Unmarshal(data, d.Value)
}

// Product and Post are structs representing two different item types.
type Product struct {
    Name  string
    Price int
}

type Post struct {
    Title   string
    Content string
}

用法:

func main() {
    // Parse a JSON item into Dynamic.
    input := `{
        "type": "product",
        "name": "iPhone",
        "price": 1000
    }`
    var dynamic Dynamic
    if err := json.Unmarshal([]byte(input), &dynamic); err != nil {
        log.Fatal(err)
    }

    // Type switch on dynamic.Value to get the parsed struct.
    // See https://tour.golang.org/methods/16
    switch dynamic.Value.(type) {
    case *Product:
        log.Println("got a product:", dynamic.Value)
    case *Post:
        log.Println("got a product:", dynamic.Value)
    }
}

输出:

2009/11/10 23:00:00得到了一个产品:&{iPhone 1000}

在Go Playground中尝试一下


提示:如果您有动态对象列表,只需将其解析为Dynamic

var items []Dynamic
json.Unmarshal(`[{...}, {...}]`, &items)

输出示例:

[&{iPhone 1000}&{A Good Post Lorem ipsum …}]

2020-07-02