我正在使用发送JSON数据的API。问题是单个元素的数组显示为单个值。例如,考虑以下JSON:
{ "names": ["Alice","Bob"] }
API将其作为数组发送。但是,当names
字段只有一个元素时,API会发送以下内容:
{ "names": "Alice" }
这是我通常在Go中解码响应的方式:
type Response struct {
Names []string `json:"names"`
}
// later
d := &Response{}
_ = json.NewDecoder(resp).Decode(d) // resp would be a http.Response.Body containing the problematic JSON
Go正确解码了第一个JSON。但是,在解码第二个JSON之后,该对象包含一个空数组。
我对API没有任何控制权,所以我必须解决这个问题。如何在Go中正确解码此JSON,以便Names
slice包含单个元素?感谢您的帮助。
您必须将其解码为interface{}
,然后使用类型断言来检查基础类型是切片还是只是字符串。
type Response struct {
Names interface{} `json:"names"`
}
然后解码为d
后,您将执行以下操作:
slice, ok := d.Names.([]string)
if ok {
// it was a slice. use it.
} else {
// it wasn't a slice - so expect it to be a string
// and use that, etc.
}
您可以实现json.Unmarshaler
接口,并让它检查json.Unmarshaler
或[
的第0个原始字节,以分别确定它是数组还是字符串:
"
type StringSlice []string
func (ss *StringSlice) UnmarshalJSON(data []byte) error {
if data[0] == '[' {
return json.Unmarshal(data, (*[]string)(ss))
} else if data[0] == '"' {
var s string
if err := json.Unmarshal(data, &s); err != nil {
return err
}
*ss = append(*ss, s)
}
return nil
}
使用https://play.golang.com/p/2GEJsS2YOLJ作为类型。 RawMessage只是延迟了部分消息的解码,因此我们以后可以自己做。
json.RawMessage
使用type Response struct {
NamesRaw json.RawMessage `json:"names"`
Names []string
}
解码json.Unmarshal
json.RawMessage