文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

如何在 Go 中将 []byte 转换为 io.Reader?

2024-12-02 11:14

关注

在 stackoverflow 上看到一个问题,题主进行了一个网络请求,接口返回的是 []byte。如果想要将其转换成 io.Reader,需要怎么做呢?

这个问题解决起来并不复杂,简单几行代码就可以轻松将其转换成功。不仅如此,还可以再通过几行代码反向转换回来。

下面听我慢慢给你吹,首先直接看两段代码。

[]byte 转 io.Reader

  1. package main 
  2.  
  3. import ( 
  4.     "bytes" 
  5.     "fmt" 
  6.     "log" 
  7.  
  8. func main() { 
  9.     data := []byte("Hello AlwaysBeta"
  10.  
  11.     // byte slice to bytes.Reader, which implements the io.Reader interface 
  12.     reader := bytes.NewReader(data) 
  13.  
  14.     // read the data from reader 
  15.     buf := make([]byte, len(data)) 
  16.     if _, err := reader.Read(buf); err != nil { 
  17.         log.Fatal(err) 
  18.     } 
  19.  
  20.     fmt.Println(string(buf)) 

输出:

  1. Hello AlwaysBeta 

这段代码先将 []byte数据转换到 reader 中,然后再从 reader 中读取数据,并打印输出。

io.Reader 转 []byte

  1. package main 
  2.  
  3. import ( 
  4.     "bytes" 
  5.     "fmt" 
  6.     "strings" 
  7.  
  8. func main() { 
  9.     ioReaderData := strings.NewReader("Hello AlwaysBeta"
  10.  
  11.     // creates a bytes.Buffer and read from io.Reader 
  12.     buf := &bytes.Buffer{} 
  13.     buf.ReadFrom(ioReaderData) 
  14.  
  15.     // retrieve a byte slice from bytes.Buffer 
  16.     data := buf.Bytes() 
  17.  
  18.     // only read the left bytes from 6 
  19.     fmt.Println(string(data[6:])) 

输出:

  1. AlwaysBeta 

这段代码先创建了一个 reader,然后读取数据到 buf,最后打印输出。

以上两段代码就是 []byte 和 io.Reader 互相转换的过程。对比这两段代码不难发现,都有 NewReader 的身影。而且在转换过程中,都起到了关键作用。

那么问题来了,这个 NewReader 到底是什么呢?接下来我们通过源码来一探究竟。

源码解析

Go 的 io 包提供了最基本的 IO 接口,其中 io.Reader 和 io.Writer 两个接口最为关键,很多原生结构都是围绕这两个接口展开的。

下面就来分别说说这两个接口:

Reader 接口

io.Reader 表示一个读取器,它将数据从某个资源读取到传输缓冲区。在缓冲区中,数据可以被流式传输和使用。

接口定义如下:

  1. type Reader interface { 
  2.     Read(p []byte) (n int, err error) 

Read() 方法将 len(p) 个字节读取到 p 中。它返回读取的字节数 n,以及发生错误时的错误信息。

举一个例子:

  1. package main 
  2.  
  3. import ( 
  4.     "fmt" 
  5.     "io" 
  6.     "os" 
  7.     "strings" 
  8.  
  9. func main() { 
  10.     reader := strings.NewReader("Clear is better than clever"
  11.     p := make([]byte, 4) 
  12.  
  13.     for { 
  14.         n, err := reader.Read(p) 
  15.         if err != nil { 
  16.             if err == io.EOF { 
  17.                 fmt.Println("EOF:", n) 
  18.                 break 
  19.             } 
  20.             fmt.Println(err) 
  21.             os.Exit(1) 
  22.         } 
  23.         fmt.Println(n, string(p[:n])) 
  24.     } 

输出:

  1. 4 Clea 
  2. 4 r is 
  3. 4  bet 
  4. 4 ter 
  5. 4 than 
  6. 4  cle 
  7. 3 ver 
  8. EOF: 0 

这段代码从 reader 不断读取数据,每次读 4 个字节,然后打印输出,直到结尾。

最后一次返回的 n 值有可能小于缓冲区大小。

Writer 接口

io.Writer 表示一个编写器,它从缓冲区读取数据,并将数据写入目标资源。

  1. type Writer interface { 
  2.    Write(p []byte) (n int, err error) 

Write 方法将 len(p) 个字节从 p 中写入到对象数据流中。它返回从 p 中被写入的字节数 n,以及发生错误时返回的错误信息。

举一个例子:

  1. package main 
  2.  
  3. import ( 
  4.     "bytes" 
  5.     "fmt" 
  6.     "os" 
  7.  
  8. func main() { 
  9.     // 创建 Buffer 暂存空间,并将一个字符串写入 Buffer 
  10.     // 使用 io.Writer 的 Write 方法写入 
  11.     var buf bytes.Buffer 
  12.     buf.Write([]byte("hello world , ")) 
  13.  
  14.     // 用 Fprintf 将一个字符串拼接到 Buffer 里 
  15.     fmt.Fprintf(&buf, " welcome to golang !"
  16.  
  17.     // 将 Buffer 的内容输出到标准输出设备 
  18.     buf.WriteTo(os.Stdout) 

输出:

  1. hello world ,  welcome to golang ! 

bytes.Buffer 是一个结构体类型,用来暂存写入的数据,其实现了 io.Writer 接口的 Write 方法。

WriteTo 方法定义:

  1. func (b *Buffer) WriteTo(w io.Writer) (n int64, err error) 

WriteTo 方法第一个参数是 io.Writer 接口类型。

转换原理

再说回文章开头的转换问题。

只要某个实例实现了接口 io.Reader 里的方法 Read() ,就满足了接口 io.Reader。

bytes 和 strings 包都实现了 Read() 方法。

  1. // src/bytes/reader.go 
  2.  
  3. // NewReader returns a new Reader reading from b. 
  4. func NewReader(b []byte) *Reader { return &Reader{b, 0, -1} } 
  5. // src/strings/reader.go 
  6.  
  7. // NewReader returns a new Reader reading from s. 
  8. // It is similar to bytes.NewBufferString but more efficient and read-only
  9. func NewReader(s string) *Reader { return &Reader{s, 0, -1} } 

 在调用 NewReader 的时候,会返回了对应的 T.Reader 类型,而它们都是通过 io.Reader 扩展而来的,所以也就实现了转换。

总结

在开发过程中,避免不了要进行一些 IO 操作,包括打印输出,文件读写,网络连接等。

在 Go 语言中,也提供了一系列标准库来应对这些操作,主要封装在以下几个包中:

除了 io.Reader 和 io.Writer 之外,io 包还封装了很多其他基本接口,比如 ReaderAt,WriterAt,ReaderFrom 和 WriterTo 等,这里就不一一介绍了。这部分代码并不复杂,读起来很轻松,而且还能加深对接口的理解,推荐大家看看。

 

来源:AlwaysBeta内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯