我正在开发一个API客户端,我需要在请求时编码JSON有效负载,并从响应中解码JSON主体。

我已经从几个库中阅读了源代码,从我所看到的,我基本上有两种编码和解码JSON字符串的可能性。

使用json。解编组传递整个响应字符串

data, err := ioutil.ReadAll(resp.Body)
if err == nil && data != nil {
    err = json.Unmarshal(data, value)
}

或者使用json.NewDecoder.Decode

err = json.NewDecoder(resp.Body).Decode(value)

在我的例子中,当处理实现io的HTTP响应时。读者,第二个版本似乎需要更少的代码,但由于我已经看到了两个版本,我想知道是否有任何偏好,我是否应该使用一个解决方案而不是另一个。

此外,这个问题的公认答案是

请使用json。解码器而不是json.Unmarshal。

但没有提到原因。我真的应该避免使用json.Unmarshal吗?


当前回答

这取决于你的输入是什么。如果你看看json的Decode方法的实现。解码器,它将整个JSON值缓冲在内存中,然后将其解组为Go值。因此,在大多数情况下,它的内存效率不会更高(尽管这在语言的未来版本中很容易改变)。

所以一个更好的经验法则是:

使用json。解码器,如果你的数据来自io。读取器流,或者需要从数据流中解码多个值。 使用json。如果内存中已经有JSON数据,则解封送。

对于从HTTP请求中读取的情况,我选择json。解码器,因为你显然是从流中读取。

其他回答

我在Go web编程书中找到了这段话。但是没有给出解释

我们什么时候使用解码器和Unmarshal?

这取决于输入。如果你的数据来自io。读者 流,就像http的正文。请求,使用解码器。如果你有 数据在字符串或内存中的某个地方,使用解组。

这取决于你的输入是什么。如果你看看json的Decode方法的实现。解码器,它将整个JSON值缓冲在内存中,然后将其解组为Go值。因此,在大多数情况下,它的内存效率不会更高(尽管这在语言的未来版本中很容易改变)。

所以一个更好的经验法则是:

使用json。解码器,如果你的数据来自io。读取器流,或者需要从数据流中解码多个值。 使用json。如果内存中已经有JSON数据,则解封送。

对于从HTTP请求中读取的情况,我选择json。解码器,因为你显然是从流中读取。