這是一個建立於 的文章,其中的資訊可能已經有所發展或是發生改變。
go語言的 io 包指定了 io.Reader 介面。go語言標準庫包含了這個介面的許多實現,包括檔案、網路連接、壓縮、加密等等。
io.Reader 介面有一個 Read 方法:
func (T) Read(b []byte) (n int, err error)
在程式中,我們使用迴圈讀取資料流,直到 error 返回 io.EOF 。
我們建立一個每次以 8 個位元組讀取 strings.Reader 的輸出的程式樣本。
package mainimport( "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 mainimport ( "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 上下載。
golang.org/x/tour/reader