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?
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.
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.
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.
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With