脚本之家,脚本语言编程技术及教程分享平台!
分类导航

Python|VBS|Ruby|Lua|perl|VBA|Golang|PowerShell|Erlang|autoit|Dos|bat|

服务器之家 - 脚本之家 - Golang - 如何在 Go 中将 []byte 转换为 io.Reader?

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

2021-12-29 23:30AlwaysBetayongxinz Golang

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

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

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

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

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

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

[]byte 转 io.Reader

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

输出:

  1. Hello AlwaysBeta

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

io.Reader 转 []byte

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

输出:

  1. AlwaysBeta

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

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

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

源码解析

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

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

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

Reader 接口

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

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

接口定义如下:

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

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

举一个例子:

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

输出:

  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 表示一个编写器,它从缓冲区读取数据,并将数据写入目标资源。

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

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

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

举一个例子:

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

输出:

  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。

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

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

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

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

总结

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

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

  • io:基本的 IO 操作接口。
  • io/ioutil:封装了一些实用的 IO 函数。
  • fmt:实现了 IO 格式化操作。
  • bufio:实现了带缓冲的 IO。
  • net.Conn:网络读写。
  • os.Stdin,os.Stdout:系统标准输入输出。
  • os.File:系统文件操作。
  • bytes:字节相关 IO 操作。

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

原文链接:https://mp.weixin.qq.com/s/nFkob92GOs6Gp75pxA5wCQ

延伸 · 阅读

精彩推荐
  • Golanggolang的httpserver优雅重启方法详解

    golang的httpserver优雅重启方法详解

    这篇文章主要给大家介绍了关于golang的httpserver优雅重启的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,...

    helight2992020-05-14
  • GolangGolang通脉之数据类型详情

    Golang通脉之数据类型详情

    这篇文章主要介绍了Golang通脉之数据类型,在编程语言中标识符就是定义的具有某种意义的词,比如变量名、常量名、函数名等等,Go语言中标识符允许由...

    4272021-11-24
  • Golanggolang 通过ssh代理连接mysql的操作

    golang 通过ssh代理连接mysql的操作

    这篇文章主要介绍了golang 通过ssh代理连接mysql的操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...

    a165861639710342021-03-08
  • Golanggolang如何使用struct的tag属性的详细介绍

    golang如何使用struct的tag属性的详细介绍

    这篇文章主要介绍了golang如何使用struct的tag属性的详细介绍,从例子说起,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看...

    Go语言中文网11352020-05-21
  • Golanggo语言制作端口扫描器

    go语言制作端口扫描器

    本文给大家分享的是使用go语言编写的TCP端口扫描器,可以选择IP范围,扫描的端口,以及多线程,有需要的小伙伴可以参考下。 ...

    脚本之家3642020-04-25
  • Golanggo日志系统logrus显示文件和行号的操作

    go日志系统logrus显示文件和行号的操作

    这篇文章主要介绍了go日志系统logrus显示文件和行号的操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...

    SmallQinYan12302021-02-02
  • GolangGolang中Bit数组的实现方式

    Golang中Bit数组的实现方式

    这篇文章主要介绍了Golang中Bit数组的实现方式,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...

    天易独尊11682021-06-09
  • Golanggolang json.Marshal 特殊html字符被转义的解决方法

    golang json.Marshal 特殊html字符被转义的解决方法

    今天小编就为大家分享一篇golang json.Marshal 特殊html字符被转义的解决方法,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧 ...

    李浩的life12792020-05-27