为了将 return 的内容 io.Reader 作为字符串指针,我最少可以复制多少次数据?

What is the smallest number of times I can copy the data in order to return the contents of an io.Reader as a string pointer?

一位同事实现了一个函数,该函数进行 HTTP 调用并将响应正文 return 发送为字符串。为简洁起见稍微简化(不,我们并没有真正忽略所有错误):

func getStuff(id string) string {
    response, _ := http.Get(fmt.Sprintf("/some/url/%s", id))
    body, _ := ioutil.ReadAll(response.Body)
    return string(body)
}

响应通常相当大,所以我想避免不必要的复制。据我了解,正如所写的那样,我们正在制作三份响应数据:

  1. io.ReadAll 将传入 HTTP 连接中的数据复制到字节片。
  2. string(body) 将字节切片复制到字符串中。
  3. return 创建字符串的新副本以供调用函数使用。

那么,首先,我对现在的状态理解正确吗?

简单的第一步是 return 一个指针:

response, _ := http.Get(fmt.Sprintf("/some/url/%s", id))
body, _ := ioutil.ReadAll(response.Body)
result := string(body)
return &result

这样就避免了第三个副本。凉爽的。但我仍在制作两份数据,我只想制作一份。

我可以让他将 return 类型更改为 *[]byte,然后我们就可以 return &body。但是所有调用者都需要自己将结果转换为字符串,然后我所做的就是将制作第二个副本的逻辑传播到其他多个地方,而不是将其合并在这里。

我可以使用 strings.Builderio.Copy:

builder := new(strings.Builder)
_, _ := io.Copy(buf, response.Body)
result buf.String()
return &result

这可能会更有效一点(我真的不知道;是吗?),但我最终还是得到了两个数据副本。

是否可以仅使用一份数据副本来完成此操作?

我认为不是;只是想知道我是否错了!

复制字符串只复制字符串头,其中包含两个词:指向包含字符串数据的数组的指针和长度。它不复制字符串内容。因此,return从函数中获取字符串不会复制该字符串。

如果您将该字符串传递给 json 解组之类的东西,您可以 return []byte,甚至 reader 来自正文,然后处理它。如果你需要它作为一个字符串,那么 two-copies 是你能拥有的最好的:一次从正文中读取它,然后将它转换成一个字符串。