go语言的io包指定了io.Reader接口。go语言标准库包含了这个接口的许多实现,包括文件、网络连接、压缩、加密等等。
    io.Reader接口有一个Read方法:

    1. func (T) Read(b []byte) (n int, err error)

    在程序中,我们使用循环读取数据流,直到error返回io.EOF。

    我们建立一个每次以8个字节读取strings.Reader的输出的程序示例

    1. package main
    2. import(
    3. "fmt"
    4. "strings"
    5. "io"
    6. )
    7. func main() {
    8. r := strings.NewReader("Hello, Reader!")
    9. b := make([]byte, 8) // 8 这里控制每次读取的字节数
    10. for{
    11. n, err := r.Read(b)
    12. fmt.Printf("n = %v err = %v b = %v\n", n, err, b)
    13. fmt.Printf("b[:n] = %q\n", b[:n])
    14. if err == io.EOF{
    15. break
    16. }
    17. }
    18. }

    运行结果

    1. n = 8 err = <nil> b = [72 101 108 108 111 44 32 82]
    2. b[:n] = "Hello, R"
    3. n = 6 err = <nil> b = [101 97 100 101 114 33 32 82]
    4. b[:n] = "eader!"
    5. n = 0 err = EOF b = [101 97 100 101 114 33 32 82]
    6. b[:n] = ""

    实现一个Reader类型,她不断生成ASCII字符’A’的流。

    1. package main
    2. import (
    3. "fmt"
    4. "io"
    5. "os"
    6. )
    7. type MyReader struct{}
    8. func Validate(r io.Reader) {
    9. b := make([]byte, 1024, 2048)
    10. i, o := 0, 0
    11. for ; i < 1<<20 && o < 1<<20; i++ { // test 1mb
    12. n, err := r.Read(b)
    13. for i, v := range b[:n] {
    14. if v != 'A' {
    15. fmt.Fprintf(os.Stderr, "got byte %x at offset %v, want 'A'\n", v, o+i)
    16. return
    17. }
    18. }
    19. o += n
    20. if err != nil {
    21. fmt.Fprintf(os.Stderr, "read error: %v\n", err)
    22. return
    23. }
    24. }
    25. if o == 0 {
    26. fmt.Fprintf(os.Stderr, "read zero bytes after %d Read calls\n", i)
    27. return
    28. }
    29. fmt.Println("OK!")
    30. }
    31. //实现一个 Reader 类型,它不断生成 ASCII 字符 'A' 的流。
    32. // TODO: Add a Read([]byte) (int, error) method to MyReader.
    33. func (mr MyReader) Read(b []byte) (n int, err error) {
    34. i := 0
    35. for ;i < len(b);i++ {
    36. b[i] = 'A'
    37. }
    38. return i, nil
    39. }
    40. func main() {
    41. Validate(MyReader{})
    42. }

    Go Readers读取数据流 - 图1