解组可能会也可能不会返回数组的JSON?

时间:2015-09-02 06:34:52

标签: json go

我正在从第三方网站(家庭电力使用)中检索JSON,并且根据我从网站请求的内容,返回的JSON可能是也可能不是数组。例如,如果我请求我的智能电表列表,我会得到这个(由于大尺寸,结果被截断):

{"gwrcmds":{"gwrcmd":{"gcmd":"SPA_UserGetSmartMeterList","gdata":{"gip":{"version":"1"...

gwrcmd 是单个元素。

但如果我要求过去半小时用电,我就明白了:

{"gwrcmds":{"gwrcmd":[{"gcmd":"DeviceGetChart","gdata":{"gip":{"version":"1" ...

了解 gwrcmd 现在是一个数组?

在我的Go应用程序中,我有一个看起来像这样的结构(再次,截断,因为它持续了一段时间。在“Version”下面有更多的子结构和属性:

type Response struct {
    Gwrcmds struct {
        Gwrcmd struct {
            Gcmd  string
            Gdata struct {
                Gip struct {
                    Version string

如果gwrcmd是一个数组,Gwrcmd必须是[]struct { },但如果不是,则只是常规的struct { }

问题是如果JSON有一个数组并且结构没有切片(反之亦然),json.Unmarshal只会返回一个错误。

我是否需要创建第二个结构复制第一个结构(除了[]struct { }之外),或者有更好的方法吗?我想到了接口的东西,但我还没有碰到那些,所以我不是百分百肯定它们。

2 个答案:

答案 0 :(得分:6)

通常,只要您拥有未知类型的JSON值,就可以使用json.RawMessage来获取它,查看它,并将其正确地解组为相应的类型。一个简化的例子:

// The A here can be either an object, or a JSON array of objects.
type Response struct {
    RawAWrapper struct {
        RawA json.RawMessage `json:"a"`
    }
    A  A   `json:"-"`
    As []A `json:"-"`
}

type A struct {
    B string
}

func (r *Response) UnmarshalJSON(b []byte) error {
    if err := json.Unmarshal(b, &r.RawAWrapper); err != nil {
        return err
    }
    if r.RawAWrapper.RawA[0] == '[' {
        return json.Unmarshal(r.RawAWrapper.RawA, &r.As)
    }
    return json.Unmarshal(r.RawAWrapper.RawA, &r.A)
}

游乐场:http://play.golang.org/p/2d_OrGltDu

根据第一个字节猜测内容对我来说似乎并不太健壮。通常,您的JSON中会有某种线索(如同动态的lengthtype字段),告诉您是否有对象或数组

另见:

答案 1 :(得分:3)

您可以尝试制作自定义json unmarshal方法,例如

func (a *GwrcmCustom) UnmarshalJSON(b []byte) (err error) {
    g, ga := Gwrcmd{}, []Gwrcmd{}
    if err = json.Unmarshal(b, &g); err == nil {
        *a = make([]Gwrcmd, 1)
        []Gwrcmd(*a)[0] = Gwrcmd(g)
        return
    }
    if err = json.Unmarshal(b, &ga); err == nil {
        *a = GwrcmCustom(ga)
        return
    }
    return
}

GwrcmCustom是自定义类型,Gwrcm

的切片
type GwrcmCustom []Gwrcmd

所以我们总是得到切片

Go playground

上试试

我希望这会有所帮助