为了账号安全,请及时绑定邮箱和手机立即绑定

从io.Reader到Go中的字符串

从io.Reader到Go中的字符串

Go
慕的地10843 2021-05-02 10:07:29
我有一个io.ReadCloser对象(来自一个http.Response对象)。将整个流转换为string对象的最有效方法是什么?
查看完整描述

3 回答

?
小怪兽爱吃肉

TA贡献1852条经验 获得超1个赞

编辑:


从1.10开始,strings.Builder存在。例子:


buf := new(strings.Builder)

n, err := io.Copy(buf, r)

// check errors

fmt.Println(buf.String())

下面的过时信息


简短的答案是,它效率不高,因为转换为字符串需要对字节数组进行完整复制。这是执行所需操作的正确方法(非有效方法):


buf := new(bytes.Buffer)

buf.ReadFrom(yourReader)

s := buf.String() // Does a complete copy of the bytes in the buffer.

此副本是作为保护机制完成的。字符串是不可变的。如果可以将[] byte转换为字符串,则可以更改字符串的内容。但是,go允许您使用unsafe软件包禁用类型安全机制。使用不安全的包装,后果自负。希望仅此名称就足够了。这是我使用不安全方法的方法:


buf := new(bytes.Buffer)

buf.ReadFrom(yourReader)

b := buf.Bytes()

s := *(*string)(unsafe.Pointer(&b))

到此为止,您现在已经有效地将字节数组转换为字符串。确实,所有这些都是在欺骗类型系统将其称为字符串。此方法有两个注意事项:

  1. 不能保证这将在所有go编译器中都有效。尽管这可以与plan-9 gc编译器一起使用,但它依赖于官方规范中未提及的“实现细节”。您甚至不能保证这将在所有体系结构上都有效,或者不能在gc中进行更改。换句话说,这是一个坏主意。

  2. 那串是可变的!如果您对该缓冲区进行任何调用,它将更改字符串。要特别小心。

我的建议是坚持官方方法。进行复制并没有那么昂贵,而且不值得进行不安全的弊端。如果字符串太大而无法进行复制,则不应将其制成字符串。


查看完整回答
反对 回复 2021-05-10
?
白衣非少年

TA贡献1155条经验 获得超0个赞

到目前为止,答案还没有解决问题的“整个流”部分。我认为这样做的好方法是ioutil.ReadAll。用你的io.ReaderCloser名字rc,我会写,


转到> = v1.16


if b, err := io.ReadAll(rc); err == nil {

    return string(b)

} ...

转到<= v1.15


if b, err := ioutil.ReadAll(rc); err == nil {

    return string(b)

} ...


查看完整回答
反对 回复 2021-05-10
  • 3 回答
  • 0 关注
  • 345 浏览
慕课专栏
更多

添加回答

举报

0/150
提交
取消
意见反馈 帮助中心 APP下载
官方微信