Decode JSON with unknown structure

前端 未结 3 1759
时光取名叫无心
时光取名叫无心 2020-12-05 01:41

I want to get a string that represents a json like this one:

{ "votes": { "option_A": "3" } }

and inclu

相关标签:
3条回答
  • 2020-12-05 02:10

    The issue I had is that sometimes I will need to get at a value that is deeply nested. Normally you would need to do a type assertion at each level, so I went ahead and just made a method that takes a map[string]interface{} and a string key, and returns the resulting map[string]interface{}.

    The issue that cropped up for me was that at some depths you will encounter a Slice instead of Map. So I also added methods to return a Slice from Map, and Map from Slice. I didnt do one for Slice to Slice, but you could easily add that if needed. Here are the methods:

    package main
    type Slice []interface{}
    type Map map[string]interface{}
    
    func (m Map) M(s string) Map {
       return m[s].(map[string]interface{})
    }
    
    func (m Map) A(s string) Slice {
       return m[s].([]interface{})
    }
    
    func (a Slice) M(n int) Map {
       return a[n].(map[string]interface{})
    }
    

    and example code:

    package main
    
    import (
       "encoding/json"
       "fmt"
       "log"
       "os"
    )
    
    func main() {
       o, e := os.Open("a.json")
       if e != nil {
          log.Fatal(e)
       }
       in_m := Map{}
       json.NewDecoder(o).Decode(&in_m)
       out_m := in_m.
          M("contents").
          M("sectionListRenderer").
          A("contents").
          M(0).
          M("musicShelfRenderer").
          A("contents").
          M(0).
          M("musicResponsiveListItemRenderer").
          M("navigationEndpoint").
          M("browseEndpoint")
       fmt.Println(out_m)
    }
    
    0 讨论(0)
  • 2020-12-05 02:15

    You really just need a single struct, and as mentioned in the comments the correct annotations on the field will yield the desired results. JSON is not some extremely variant data format, it is well defined and any piece of json, no matter how complicated and confusing it might be to you can be represented fairly easily and with 100% accuracy both by a schema and in objects in Go and most other OO programming languages. Here's an example;

    package main
    
    import (
        "fmt"
        "encoding/json"
    )
    
    type Data struct {
        Votes *Votes `json:"votes"`
        Count string `json:"count,omitempty"`
    }
    
    type Votes struct {
        OptionA string `json:"option_A"`
    }
    
    func main() {
        s := `{ "votes": { "option_A": "3" } }`
        data := &Data{
            Votes: &Votes{},
        }
        err := json.Unmarshal([]byte(s), data)
        fmt.Println(err)
        fmt.Println(data.Votes)
        s2, _ := json.Marshal(data)
        fmt.Println(string(s2))
        data.Count = "2"
        s3, _ := json.Marshal(data)
        fmt.Println(string(s3))
    }
    

    https://play.golang.org/p/ScuxESTW5i

    Based on your most recent comment you could address that by using an interface{} to represent data besides the count, making the count a string and having the rest of the blob shoved into the interface{} which will accept essentially anything. That being said, Go is a statically typed language with a fairly strict type system and to reiterate, your comments stating 'it can be anything' are not true. JSON cannot be anything. For any piece of JSON there is schema and a single schema can define many many variations of JSON. I advise you take the time to understand the structure of your data rather than hacking something together under the notion that it cannot be defined when it absolutely can and is probably quite easy for someone who knows what they're doing.

    0 讨论(0)
  • 2020-12-05 02:31
    package main
    
    import "encoding/json"
    
    func main() {
        in := []byte(`{ "votes": { "option_A": "3" } }`)
        var raw map[string]interface{}
        if err := json.Unmarshal(in, &raw); err != nil {
            panic(err)
        }
        raw["count"] = 1
        out, err := json.Marshal(raw)
        if err != nil {
            panic(err)
        }
        println(string(out))
    }
    

    https://play.golang.org/p/o8ZwvgsQmoO

    0 讨论(0)
提交回复
热议问题