go語言的io包指定了io.Reader接口。go語言标準庫包含了這個接口的許多實作,包括檔案、網絡連接配接、壓縮、加密等等。
io.Reader接口有一個Read方法:
func (T) Read(b []byte) (n int, err error)
在程式中,我們使用循環讀取資料流,直到error傳回io.EOF。
我們建立一個每次以8個位元組讀取strings.Reader的輸出的程式示例
package main
import(
"fmt"
"strings"
"io"
)
func main() {
r := strings.NewReader("Hello, Reader!")
b := make([]byte, 8) // 8 這裡控制每次讀取的位元組數
for{
n, err := r.Read(b)
fmt.Printf("n = %v err = %v b = %v\n", n, err, b)
fmt.Printf("b[:n] = %q\n", b[:n])
if err == io.EOF{
break
}
}
}
運作結果
n = 8 err = <nil> b = [72 101 108 108 111 44 32 82]
b[:n] = "Hello, R"
n = 6 err = <nil> b = [101 97 100 101 114 33 32 82]
b[:n] = "eader!"
n = 0 err = EOF b = [101 97 100 101 114 33 32 82]
b[:n] = ""
實作一個Reader類型,她不斷生成ASCII字元'A'的流。
package main
import (
"fmt"
"io"
"os"
)
type MyReader struct{}
func Validate(r io.Reader) {
b := make([]byte, 1024, 2048)
i, o := 0, 0
for ; i < 1<<20 && o < 1<<20; i++ { // test 1mb
n, err := r.Read(b)
for i, v := range b[:n] {
if v != 'A' {
fmt.Fprintf(os.Stderr, "got byte %x at offset %v, want 'A'\n", v, o+i)
return
}
}
o += n
if err != nil {
fmt.Fprintf(os.Stderr, "read error: %v\n", err)
return
}
}
if o == 0 {
fmt.Fprintf(os.Stderr, "read zero bytes after %d Read calls\n", i)
return
}
fmt.Println("OK!")
}
//實作一個 Reader 類型,它不斷生成 ASCII 字元 'A' 的流。
// TODO: Add a Read([]byte) (int, error) method to MyReader.
func (mr MyReader) Read(b []byte) (n int, err error) {
i := 0
for ;i < len(b);i++ {
b[i] = 'A'
}
return i, nil
}
func main() {
Validate(MyReader{})
}
代碼中的 func Validate(r io.Reader) 可以從 github 上下載下傳。
Github: https://github.com/golang/tour
參考:
https://www.yuque.com/docs/share/0952086f-f2b6-401d-bcb5-0a9aa9fffec8