动态JSON结构,API结果golang

时间:2018-08-18 00:34:17

标签: json api http parsing go

我必须在GoLang中进行两个HTTP API调用,第一个API调用返回此json响应:

{
  "status": 200,
  "msg": "OK",
  "result": {
    "id": "24",
    "folderid": "4248"
  }
}

我的第一个响应的json结构是这样设置的:

type One struct {
    Status int    `json:"status"`
    Msg    string `json:"msg"`
    Result struct {
        ID       string `json:"id"`
        Folderid string `json:"folderid"`
    } `json:"result"`
}

第二个电话是问题所在。如您所见,第一个API调用返回一个结果-> id。该ID应该是第二个结构的开头的名称,但我似乎无法使它动态化或将结果作为结构名称。该ID(24)将始终根据第一个API调用而更改。我目前无法解析第二个调用的JSON并设置我的结构。在第二次API调用中,我要访问remoteurl / status。

第二次通话结果(我无法解析):

{
  "status": 200,
  "msg": "OK",
  "result": {
    24: ** THIS IS DYNAMIC** {
      "id": 24,
      "remoteurl": "http://proof.ovh.net/files/100Mio.dat",
      "status": "new",
      "bytes_loaded": null,
      "bytes_total": null,
      "folderid": "4248",
      "added": "2015-02-21 09:20:26",
      "last_update": "2015-02-21 09:20:26",
      "extid": false,
      "url": false
    }
  }
}

有人知道如何设置我的结构或解决此问题。我是一名新程序员,并且为此工作了4天。并决定寻求帮助,因为我在学校并且已完成正常的作业。

1 个答案:

答案 0 :(得分:1)

{
  "status": 200,
  "msg": "OK",
  "result": {
    24:  {
      "id": 24,
      "remoteurl": "http://proof.ovh.net/files/100Mio.dat",
      "status": "new",
      "bytes_loaded": null,
      "bytes_total": null,
      "folderid": "4248",
      "added": "2015-02-21 09:20:26",
      "last_update": "2015-02-21 09:20:26",
      "extid": false,
      "url": false
    }
  }
}

不值JSON。您必须指的是我在下面发布的JSON,如果您想检查一下自己,请将您的JSON版本复制到任何JSON验证程序中;

https://jsonlint.com/

https://jsoneditoronline.org/

https://jsonformatter.curiousconcept.com/

还请查看下面链接的线程。.如果API确实返回了您声称返回的内容,则说明API中存在错误

Why JSON allows only string to be a key?

{
  "status": 200,
  "msg": "OK",
  "result": {
    "24":  {
      "id": 24,
      "remoteurl": "http://proof.ovh.net/files/100Mio.dat",
      "status": "new",
      "bytes_loaded": null,
      "bytes_total": null,
      "folderid": "4248",
      "added": "2015-02-21 09:20:26",
      "last_update": "2015-02-21 09:20:26",
      "extid": false,
      "url": false
    }
  }
}

下面是一些示例代码,该示例使用映射到结构来解决第二个响应的动态响应

package main

import (
    "encoding/json"
    "fmt"
    "log"
)

var res1 = `{
  "status": 200,
  "msg": "OK",
  "result": {
    "id": "24",
    "folderid": "4248"
  }
}`

var res2 = `{
  "status": 200,
  "msg": "OK",
  "result": {
    "24":  {
      "id": 24,
      "remoteurl": "http://proof.ovh.net/files/100Mio.dat",
      "status": "new",
      "bytes_loaded": null,
      "bytes_total": null,
      "folderid": "4248",
      "added": "2015-02-21 09:20:26",
      "last_update": "2015-02-21 09:20:26",
      "extid": false,
      "url": false
    }
  }
}
`

type One struct {
    Status int    `json:"status"`
    Msg    string `json:"msg"`
    Result struct {
        ID       string `json:"id"`
        Folderid string `json:"folderid"`
    } `json:"result"`
}

type Two struct {
    Status int                  `json:"status"`
    Msg    string               `json:"msg"`
    Result map[string]innerData `json:"result"`
}

type innerData struct {
    ID          int         `json:"id"`
    Remoteurl   string      `json:"remoteurl"`
    Status      string      `json:"status"`
    BytesLoaded interface{} `json:"bytes_loaded"`
    BytesTotal  interface{} `json:"bytes_total"`
    Folderid    string      `json:"folderid"`
    Added       string      `json:"added"`
    LastUpdate  string      `json:"last_update"`
    Extid       bool        `json:"extid"`
    URL         bool        `json:"url"`
}

func main() {
    var one One
    err := json.Unmarshal([]byte(res1), &one)
    if err != nil {
        log.Fatal(err)
    }

    var two Two
    err = json.Unmarshal([]byte(res2), &two)
    if err != nil {
        log.Fatal(err)
    }

    //pretty print both strutures
    b, _ := json.MarshalIndent(one, "", " ")
    fmt.Printf("%s \n\n", b)
    b, _ = json.MarshalIndent(two, "", " ")
    fmt.Printf("%s \n\n", b)

    // access data from two with id from one
    if dat, ok := two.Result[one.Result.ID]; ok {
        b, _ = json.MarshalIndent(dat, "", " ")
        fmt.Printf("inner data\n%s\n", b)
    }

}