如何在Go中將[]byte轉(zhuǎn)換為io.Reader
在 stackoverflow 上看到一個(gè)問(wèn)題,題主進(jìn)行了一個(gè)網(wǎng)絡(luò)請(qǐng)求,接口返回的是 []byte。如果想要將其轉(zhuǎn)換成 io.Reader,需要怎么做呢?
這個(gè)問(wèn)題解決起來(lái)并不復(fù)雜,簡(jiǎn)單幾行代碼就可以輕松將其轉(zhuǎn)換成功。不僅如此,還可以再通過(guò)幾行代碼反向轉(zhuǎn)換回來(lái)。
下面聽(tīng)我慢慢給你吹,首先直接看兩段代碼。
[]byte 轉(zhuǎn) 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 數(shù)據(jù)轉(zhuǎn)換到 reader 中,然后再?gòu)?reader 中讀取數(shù)據(jù),并打印輸出。
io.Reader 轉(zhuǎn) []byte
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
這段代碼先創(chuàng)建了一個(gè) reader,然后讀取數(shù)據(jù)到 buf,最后打印輸出。
以上兩段代碼就是 []byte 和 io.Reader 互相轉(zhuǎn)換的過(guò)程。對(duì)比這兩段代碼不難發(fā)現(xiàn),都有 NewReader 的身影。而且在轉(zhuǎn)換過(guò)程中,都起到了關(guān)鍵作用。
那么問(wèn)題來(lái)了,這個(gè) NewReader 到底是什么呢?接下來(lái)我們通過(guò)源碼來(lái)一探究竟。
源碼解析
Go 的 io 包提供了最基本的 IO 接口,其中 io.Reader 和 io.Writer 兩個(gè)接口最為關(guān)鍵,很多原生結(jié)構(gòu)都是圍繞這兩個(gè)接口展開(kāi)的。
下面就來(lái)分別說(shuō)說(shuō)這兩個(gè)接口:
Reader 接口
io.Reader 表示一個(gè)讀取器,它將數(shù)據(jù)從某個(gè)資源讀取到傳輸緩沖區(qū)。在緩沖區(qū)中,數(shù)據(jù)可以被流式傳輸和使用。
接口定義如下:
type Reader interface { Read(p []byte) (n int, err error) }
Read() 方法將 len(p) 個(gè)字節(jié)讀取到 p 中。它返回讀取的字節(jié)數(shù) n,以及發(fā)生錯(cuò)誤時(shí)的錯(cuò)誤信息。
舉一個(gè)例子:
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 不斷讀取數(shù)據(jù),每次讀 4 個(gè)字節(jié),然后打印輸出,直到結(jié)尾。
最后一次返回的 n 值有可能小于緩沖區(qū)大小。
Writer 接口
io.Writer 表示一個(gè)編寫(xiě)器,它從緩沖區(qū)讀取數(shù)據(jù),并將數(shù)據(jù)寫(xiě)入目標(biāo)資源。
type Writer interface { Write(p []byte) (n int, err error) }
Write 方法將 len(p) 個(gè)字節(jié)從 p 中寫(xiě)入到對(duì)象數(shù)據(jù)流中。它返回從 p 中被寫(xiě)入的字節(jié)數(shù) n,以及發(fā)生錯(cuò)誤時(shí)返回的錯(cuò)誤信息。
舉一個(gè)例子:
package main import ( "bytes" "fmt" "os" ) func main() { // 創(chuàng)建 Buffer 暫存空間,并將一個(gè)字符串寫(xiě)入 Buffer // 使用 io.Writer 的 Write 方法寫(xiě)入 var buf bytes.Buffer buf.Write([]byte("hello world , ")) // 用 Fprintf 將一個(gè)字符串拼接到 Buffer 里 fmt.Fprintf(&buf, " welcome to golang !") // 將 Buffer 的內(nèi)容輸出到標(biāo)準(zhǔn)輸出設(shè)備 buf.WriteTo(os.Stdout) }
輸出:
hello world ,? welcome to golang !
bytes.Buffer 是一個(gè)結(jié)構(gòu)體類(lèi)型,用來(lái)暫存寫(xiě)入的數(shù)據(jù),其實(shí)現(xiàn)了 io.Writer 接口的 Write 方法。
WriteTo 方法定義:
func (b *Buffer) WriteTo(w io.Writer) (n int64, err error)
WriteTo 方法第一個(gè)參數(shù)是 io.Writer 接口類(lèi)型。
轉(zhuǎn)換原理
再說(shuō)回文章開(kāi)頭的轉(zhuǎn)換問(wèn)題。
只要某個(gè)實(shí)例實(shí)現(xiàn)了接口 io.Reader 里的方法 Read() ,就滿足了接口 io.Reader。
bytes 和 strings 包都實(shí)現(xiàn)了 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} }
在調(diào)用 NewReader 的時(shí)候,會(huì)返回了對(duì)應(yīng)的 T.Reader 類(lèi)型,而它們都是通過(guò) io.Reader 擴(kuò)展而來(lái)的,所以也就實(shí)現(xiàn)了轉(zhuǎn)換。
總結(jié)
在開(kāi)發(fā)過(guò)程中,避免不了要進(jìn)行一些 IO 操作,包括打印輸出,文件讀寫(xiě),網(wǎng)絡(luò)連接等。
在 Go 語(yǔ)言中,也提供了一系列標(biāo)準(zhǔn)庫(kù)來(lái)應(yīng)對(duì)這些操作,主要封裝在以下幾個(gè)包中:
- io:基本的 IO 操作接口。
- io/ioutil:封裝了一些實(shí)用的 IO 函數(shù)。
- fmt:實(shí)現(xiàn)了 IO 格式化操作。
- bufio:實(shí)現(xiàn)了帶緩沖的 IO。
- net.Conn:網(wǎng)絡(luò)讀寫(xiě)。
- 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ù)雜,讀起來(lái)很輕松,而且還能加深對(duì)接口的理解,推薦大家看看。
到此這篇關(guān)于如何在Go中將[]byte轉(zhuǎn)換為io.Reader的文章就介紹到這了,更多相關(guān)Go []byte轉(zhuǎn)換為io.Reader內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Go語(yǔ)言題解LeetCode1260二維網(wǎng)格遷移示例詳解
這篇文章主要為大家介紹了Go語(yǔ)言題解LeetCode1260二維網(wǎng)格遷移示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-01-01Golang使用pprof和trace進(jìn)行診斷和修復(fù)性能問(wèn)題
在?Go?中,開(kāi)發(fā)人員可以使用強(qiáng)大的內(nèi)置工具來(lái)幫助診斷和修復(fù)性能問(wèn)題,其中兩個(gè)工具是?pprof?和?trace?包,下面就跟隨小編一起來(lái)了解下如何使用pprof和trace進(jìn)行診斷和修復(fù)性能問(wèn)題吧2024-01-01golang 如何用反射reflect操作結(jié)構(gòu)體
這篇文章主要介紹了golang 用反射reflect操作結(jié)構(gòu)體的操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2021-04-04go json數(shù)據(jù)轉(zhuǎn)發(fā)的實(shí)現(xiàn)代碼
這篇文章主要介紹了go json數(shù)據(jù)轉(zhuǎn)發(fā)的實(shí)現(xiàn)代碼,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-09-09go責(zé)任鏈行為型設(shè)計(jì)模式Chain?Of?Responsibility
這篇文章主要為大家介紹了go行為型設(shè)計(jì)模式之責(zé)任鏈Chain?Of?Responsibility使用示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-12-12Golang定制化zap日志庫(kù)使用過(guò)程分析
Zap是我個(gè)人比較喜歡的日志庫(kù),是uber開(kāi)源的,有較好的性能,在項(xiàng)目開(kāi)發(fā)中,經(jīng)常需要把程序運(yùn)行過(guò)程中各種信息記錄下來(lái),有了詳細(xì)的日志有助于問(wèn)題排查和功能優(yōu)化,但如何選擇和使用性能好功能強(qiáng)大的日志庫(kù),這個(gè)就需要我們從多角度考慮2023-03-03go內(nèi)存緩存BigCache封裝Entry源碼解讀
這篇文章主要為大家介紹了go內(nèi)存緩存BigCache封裝Entry源碼解讀,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-09-09Go語(yǔ)言輕量級(jí)線程Goroutine用法實(shí)例
這篇文章主要介紹了Go語(yǔ)言輕量級(jí)線程Goroutine用法,實(shí)例分析了goroutine使用技巧,需要的朋友可以參考下2015-02-02