Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

What's the golang equivalent of converting any JSON to standard dict in Python?

In Python you can do something like this:

r = requests.get("http://wikidata.org/w/api.php", params=params)
data = r.json()

And now data is a dict or hash table (also, I did not need to define beforehand the structure of the dict), and I can access values of keys by doing data["entities"], data["entities"]["Q12"], etc.

How would I do this in golang? So far I have this:

resp, err := http.Get("http://wikidata.org/w/api.php?"+v.Encode())
if err != nil {
    // handle error
}

defer resp.Body.Close()
decoder := json.NewDecoder(resp.Body)
var data interface{}
decodeErr := decoder.Decode(&data)
if decodeErr != nil {
    // handle error
}
fmt.Println(data["entities"], data["entities"]["Q"+id])

Which gives me the compile error: invalid operation: data["entities"] (index of type interface {})

So what type should var data be? And do I need to define a structure to the JSON beforehand or is it possible to handle any JSON file/stream without modifying the code?

like image 764
01AutoMonkey Avatar asked Mar 04 '15 16:03

01AutoMonkey


People also ask

Is JSON format same as Python dictionary?

oranges comparison: JSON is a data format (a string), Python dictionary is a data structure (in-memory object). If you need to exchange data between different (perhaps even non-Python) processes then you could use JSON format to serialize your Python dictionary.

Which function will convert the Python dictionary to valid JSON?

The json. dumps() is a built-in Python function that converts the dictionary to a string object. The “json” module makes it easy to parse the JSON strings which contain the JSON object.


1 Answers

If you want a dictionary, use the Go type map[string]interface{} (which is a map with string keys and values of any type):

var data map[string]interface{}

And then you can refer to its elements like:

data["entities"]

See this example:

s := `{"text":"I'm a text.","number":1234,"floats":[1.1,2.2,3.3],
    "innermap":{"foo":1,"bar":2}}`

var data map[string]interface{}
err := json.Unmarshal([]byte(s), &data)
if err != nil {
    panic(err)
}

fmt.Println("text =", data["text"])
fmt.Println("number =", data["number"])
fmt.Println("floats =", data["floats"])
fmt.Println("innermap =", data["innermap"])

innermap, ok := data["innermap"].(map[string]interface{})
if !ok {
    panic("inner map is not a map!")
}
fmt.Println("innermap.foo =", innermap["foo"])
fmt.Println("innermap.bar =", innermap["bar"])

fmt.Println("The whole map:", data)

Output:

text = I'm a text.
number = 1234
floats = [1.1 2.2 3.3]
innermap = map[foo:1 bar:2]
innermap.foo = 1
innermap.bar = 2
The whole map: map[text:I'm a text. number:1234 floats:[1.1 2.2 3.3]
    innermap:map[foo:1 bar:2]]

Try it on the Go Playground.

Notes:

Basically if your map is multi-level (the map contains another map) like the "innermap" in the above example, when you access the inner map, you can use Type assertion to have it as another map:

innermap, ok := data["innermap"].(map[string]interface{})
// If ok, innermap is of type map[string]interface{}
// and you can refer to its elements.
like image 193
icza Avatar answered Sep 18 '22 01:09

icza