Golang -返回的JSON顶级字段是可变的,如何在结构中使用

o2gm4chl  于 4个月前  发布在  Go
关注(0)|答案(1)|浏览(68)

我有一个来自API的JSON响应,看起来像这样:

{
    "1091500": {
      "data": {
        "price_overview": {
          "final_formatted": "$59.99"
        }
      } 
    }
}

字符串
当我解组并访问我需要的价格值时,我的结构体工作得很好。price.AppID.Data.PriceOverview.FinalFormatted
struct在这里:

type SteamApp struct {
    AppID struct {
        Data struct {
            PriceOverview struct {
                FinalFormatted string `json:"final_formatted"`
            } `json:"price_overview"`
        } `json:"data"`
    } `json:"1091500"`
}


我尝试处理的是顶级键,“1091500”。这是一个由单独的函数生成的值,实际上是抓取上述数据所需的搜索项。也就是说,它是动态的,在运行时由单独的API调用发现。因为您需要在结构中使用JSON标记,所以我很难想出一种方法来泛化它。
到目前为止,最好的方法是找到一个不同的API,以不同的方式返回数据,但我还没有找到。我还想找到一种方法来“跳过”最初的部分,因为我有正确的数据,这只是它的“阅读”的问题。提前谢谢你,我希望我解释得足够好。

2exbekwf

2exbekwf1#

分两步进行解组。首先是Map,然后是结构类型。你可以通过实现json自动完成。解组器:

package main

import (
    "encoding/json"
    "fmt"
)

var j = []byte(`{
    "1091500": {
      "data": {
        "price_overview": {
          "final_formatted": "$59.99"
        }
      }
    }
}`)

func main() {
    var t T
    err := json.Unmarshal(j, &t)

    fmt.Println(err)
    fmt.Printf("%+v\n", t)
}

type T struct {
    AppID         string `json:"-"`
    PriceOverview struct {
        FileFormatted string `json:"final_formatted"`
    } `json:"price_overview"`
}

func (t *T) UnmarshalJSON(b []byte) error {
    var m map[string]struct {
        Data json.RawMessage
    }

    err := json.Unmarshal(b, &m)
    if err != nil {
        return err
    }

    type T_ T // prevent recursion
    var t_ T_

    for appID, x := range m {
        err := json.Unmarshal(x.Data, &t_)
        if err != nil {
            return err
        }

        *t = T(t_)
        t.AppID = appID

        break
    }

    return nil
}

字符串
在操场上试试:https://go.dev/play/p/NMffrAWEBQy
我假设data属性只是噪音,并使其透明。如果我错了,那么使用map[string] json.RawMessage并将Data字段添加为类型T。
如果你想自动编组成相同的格式,也可以实现json.Marshaler:

func (t T) MarshalJSON() ([]byte, error) {
    type T_ T // prevent recursion

    m := map[string]struct {
        Data T_ `json:"data"`
    }{
        t.AppID: {Data: T_(t)},
    }

    return json.Marshal(m)
}

相关问题