您好,登錄后才能下訂單哦!
這篇文章主要介紹怎么在Go中將[]byte轉換為io.Reader,文中介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們一定要看完!
在 stackoverflow 上看到一個問題,題主進行了一個網絡請求,接口返回的是 []byte。如果想要將其轉換成 io.Reader,需要怎么做呢?
這個問題解決起來并不復雜,簡單幾行代碼就可以輕松將其轉換成功。不僅如此,還可以再通過幾行代碼反向轉換回來。
下面聽我慢慢給你吹,首先直接看兩段代碼。
package main import ( "bytes" "fmt" "log" ) func main() { data := []byte("Hello AlwaysBeta") // byte slice to bytes.Reader, which implements the io.Reader interface reader := bytes.NewReader(data) // read the data from reader buf := make([]byte, len(data)) if _, err := reader.Read(buf); err != nil { log.Fatal(err) } fmt.Println(string(buf)) }
輸出:
Hello AlwaysBeta
這段代碼先將 []byte 數據轉換到 reader 中,然后再從 reader 中讀取數據,并打印輸出。
package main import ( "bytes" "fmt" "strings" ) func main() { ioReaderData := strings.NewReader("Hello AlwaysBeta") // creates a bytes.Buffer and read from io.Reader buf := &bytes.Buffer{} buf.ReadFrom(ioReaderData) // retrieve a byte slice from bytes.Buffer data := buf.Bytes() // only read the left bytes from 6 fmt.Println(string(data[6:])) }
輸出:
AlwaysBeta
這段代碼先創建了一個 reader,然后讀取數據到 buf,最后打印輸出。
以上兩段代碼就是 []byte 和 io.Reader 互相轉換的過程。對比這兩段代碼不難發現,都有 NewReader 的身影。而且在轉換過程中,都起到了關鍵作用。
那么問題來了,這個 NewReader 到底是什么呢?接下來我們通過源碼來一探究竟。
Go 的 io 包提供了最基本的 IO 接口,其中 io.Reader 和 io.Writer 兩個接口最為關鍵,很多原生結構都是圍繞這兩個接口展開的。
下面就來分別說說這兩個接口:
io.Reader 表示一個讀取器,它將數據從某個資源讀取到傳輸緩沖區。在緩沖區中,數據可以被流式傳輸和使用。
接口定義如下:
type Reader interface { Read(p []byte) (n int, err error) }
Read() 方法將 len(p) 個字節讀取到 p 中。它返回讀取的字節數 n,以及發生錯誤時的錯誤信息。
舉一個例子:
package main import ( "fmt" "io" "os" "strings" ) func main() { reader := strings.NewReader("Clear is better than clever") p := make([]byte, 4) for { n, err := reader.Read(p) if err != nil { if err == io.EOF { fmt.Println("EOF:", n) break } fmt.Println(err) os.Exit(1) } fmt.Println(n, string(p[:n])) } }
輸出:
4 Clea
4 r is
4 bet
4 ter
4 than
4 cle
3 ver
EOF: 0
這段代碼從 reader 不斷讀取數據,每次讀 4 個字節,然后打印輸出,直到結尾。
最后一次返回的 n 值有可能小于緩沖區大小。
io.Writer 表示一個編寫器,它從緩沖區讀取數據,并將數據寫入目標資源。
type Writer interface { Write(p []byte) (n int, err error) }
Write 方法將 len(p) 個字節從 p 中寫入到對象數據流中。它返回從 p 中被寫入的字節數 n,以及發生錯誤時返回的錯誤信息。
舉一個例子:
package main import ( "bytes" "fmt" "os" ) func main() { // 創建 Buffer 暫存空間,并將一個字符串寫入 Buffer // 使用 io.Writer 的 Write 方法寫入 var buf bytes.Buffer buf.Write([]byte("hello world , ")) // 用 Fprintf 將一個字符串拼接到 Buffer 里 fmt.Fprintf(&buf, " welcome to golang !") // 將 Buffer 的內容輸出到標準輸出設備 buf.WriteTo(os.Stdout) }
輸出:
hello world , welcome to golang !
bytes.Buffer 是一個結構體類型,用來暫存寫入的數據,其實現了 io.Writer 接口的 Write 方法。
WriteTo 方法定義:
func (b *Buffer) WriteTo(w io.Writer) (n int64, err error)
WriteTo 方法第一個參數是 io.Writer 接口類型。
再說回文章開頭的轉換問題。
只要某個實例實現了接口 io.Reader 里的方法 Read() ,就滿足了接口 io.Reader。
bytes 和 strings 包都實現了 Read() 方法。
// src/bytes/reader.go // NewReader returns a new Reader reading from b. func NewReader(b []byte) *Reader { return &Reader{b, 0, -1} }
// src/strings/reader.go // NewReader returns a new Reader reading from s. // It is similar to bytes.NewBufferString but more efficient and read-only. func NewReader(s string) *Reader { return &Reader{s, 0, -1} }
在調用 NewReader 的時候,會返回了對應的 T.Reader 類型,而它們都是通過 io.Reader 擴展而來的,所以也就實現了轉換。
以上是“怎么在Go中將[]byte轉換為io.Reader”這篇文章的所有內容,感謝各位的閱讀!希望分享的內容對大家有幫助,更多相關知識,歡迎關注億速云行業資訊頻道!
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。