自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

如何在 Go 中將 []byte 轉(zhuǎn)換為 io.Reader?

開發(fā) 前端
在 stackoverflow 上看到一個問題,題主進(jìn)行了一個網(wǎng)絡(luò)請求,接口返回的是 []byte。如果想要將其轉(zhuǎn)換成 io.Reader,需要怎么做呢?

[[442722]]

在 stackoverflow 上看到一個問題,題主進(jìn)行了一個網(wǎng)絡(luò)請求,接口返回的是 []byte。如果想要將其轉(zhuǎn)換成 io.Reader,需要怎么做呢?

這個問題解決起來并不復(fù)雜,簡單幾行代碼就可以輕松將其轉(zhuǎn)換成功。不僅如此,還可以再通過幾行代碼反向轉(zhuǎn)換回來。

下面聽我慢慢給你吹,首先直接看兩段代碼。

[]byte 轉(zhuǎn) 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數(shù)據(jù)轉(zhuǎn)換到 reader 中,然后再從 reader 中讀取數(shù)據(jù),并打印輸出。

io.Reader 轉(zhuǎn) []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 

這段代碼先創(chuàng)建了一個 reader,然后讀取數(shù)據(jù)到 buf,最后打印輸出。

以上兩段代碼就是 []byte 和 io.Reader 互相轉(zhuǎn)換的過程。對比這兩段代碼不難發(fā)現(xiàn),都有 NewReader 的身影。而且在轉(zhuǎn)換過程中,都起到了關(guān)鍵作用。

那么問題來了,這個 NewReader 到底是什么呢?接下來我們通過源碼來一探究竟。

源碼解析

Go 的 io 包提供了最基本的 IO 接口,其中 io.Reader 和 io.Writer 兩個接口最為關(guān)鍵,很多原生結(jié)構(gòu)都是圍繞這兩個接口展開的。

下面就來分別說說這兩個接口:

Reader 接口

io.Reader 表示一個讀取器,它將數(shù)據(jù)從某個資源讀取到傳輸緩沖區(qū)。在緩沖區(qū)中,數(shù)據(jù)可以被流式傳輸和使用。

接口定義如下:

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

Read() 方法將 len(p) 個字節(jié)讀取到 p 中。它返回讀取的字節(jié)數(shù) n,以及發(fā)生錯誤時的錯誤信息。

舉一個例子:

  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 不斷讀取數(shù)據(jù),每次讀 4 個字節(jié),然后打印輸出,直到結(jié)尾。

最后一次返回的 n 值有可能小于緩沖區(qū)大小。

Writer 接口

io.Writer 表示一個編寫器,它從緩沖區(qū)讀取數(shù)據(jù),并將數(shù)據(jù)寫入目標(biāo)資源。

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

Write 方法將 len(p) 個字節(jié)從 p 中寫入到對象數(shù)據(jù)流中。它返回從 p 中被寫入的字節(jié)數(shù) n,以及發(fā)生錯誤時返回的錯誤信息。

舉一個例子:

  1. package main 
  2.  
  3. import ( 
  4.     "bytes" 
  5.     "fmt" 
  6.     "os" 
  7.  
  8. func main() { 
  9.     // 創(chuàng)建 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 的內(nèi)容輸出到標(biāo)準(zhǔn)輸出設(shè)備 
  18.     buf.WriteTo(os.Stdout) 

輸出:

  1. hello world ,  welcome to golang ! 

bytes.Buffer 是一個結(jié)構(gòu)體類型,用來暫存寫入的數(shù)據(jù),其實現(xiàn)了 io.Writer 接口的 Write 方法。

WriteTo 方法定義:

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

WriteTo 方法第一個參數(shù)是 io.Writer 接口類型。

轉(zhuǎn)換原理

再說回文章開頭的轉(zhuǎn)換問題。

只要某個實例實現(xiàn)了接口 io.Reader 里的方法 Read() ,就滿足了接口 io.Reader。

bytes 和 strings 包都實現(xiàn)了 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} } 

 在調(diào)用 NewReader 的時候,會返回了對應(yīng)的 T.Reader 類型,而它們都是通過 io.Reader 擴(kuò)展而來的,所以也就實現(xiàn)了轉(zhuǎn)換。

總結(jié)

在開發(fā)過程中,避免不了要進(jìn)行一些 IO 操作,包括打印輸出,文件讀寫,網(wǎng)絡(luò)連接等。

在 Go 語言中,也提供了一系列標(biāo)準(zhǔn)庫來應(yīng)對這些操作,主要封裝在以下幾個包中:

  • io:基本的 IO 操作接口。
  • io/ioutil:封裝了一些實用的 IO 函數(shù)。
  • fmt:實現(xiàn)了 IO 格式化操作。
  • bufio:實現(xiàn)了帶緩沖的 IO。
  • net.Conn:網(wǎng)絡(luò)讀寫。
  • os.Stdin,os.Stdout:系統(tǒng)標(biāo)準(zhǔn)輸入輸出。
  • os.File:系統(tǒng)文件操作。
  • bytes:字節(jié)相關(guān) IO 操作。

除了 io.Reader 和 io.Writer 之外,io 包還封裝了很多其他基本接口,比如 ReaderAt,WriterAt,ReaderFrom 和 WriterTo 等,這里就不一一介紹了。這部分代碼并不復(fù)雜,讀起來很輕松,而且還能加深對接口的理解,推薦大家看看。

 

責(zé)任編輯:姜華 來源: AlwaysBeta
相關(guān)推薦

2025-02-12 11:06:24

Go函數(shù)MultiBytes

2024-07-09 08:07:37

Go性能工具

2020-04-06 20:30:37

JavaScriptBoolean開發(fā)

2022-12-25 16:15:38

HTMLJava可視化文檔

2022-12-25 15:56:30

JavaScript字符串

2016-11-15 14:29:14

Linux文件編碼轉(zhuǎn)換

2015-11-24 09:53:22

AngularJSXMLJSON

2022-10-20 08:59:18

Go接口類型

2024-09-18 08:00:05

C#編程

2021-10-29 15:13:21

LinuxPDF文件

2024-03-12 07:35:39

Python字符串列表

2023-11-07 09:02:07

Golangbytes

2021-06-07 12:20:14

LinuxASCII命令

2023-08-28 17:16:51

Golangio 包

2015-09-14 14:49:39

MySQLMariaDBLinux

2022-09-22 11:40:11

JavaScript數(shù)組開發(fā)

2023-01-28 08:05:32

轉(zhuǎn)換Go泛型

2019-03-28 08:00:00

Linux磁盤IO監(jiān)控存儲設(shè)備

2022-05-31 10:38:50

Linux密碼scp

2021-07-09 12:37:31

GoPython編程語言
點贊
收藏

51CTO技術(shù)棧公眾號