本教程详解go语言官方tour中第22题——如何正确实现`io.reader`接口,通过覆写`read([]byte) (int, error)`方法,将传入字节切片的所有位置填充为ascii字符'a',从而模拟一个无限流式数据源。
在Go语言中,io.Reader 是一个核心接口,定义如下:
type Reader interface {
Read(p []byte) (n int, err error)
}其 Read 方法不负责分配内存,也不返回新切片,而是要求你就地修改传入的字节切片 p,填入尽可能多的有效数据,并返回实际写入的字节数
n(通常为 len(p))以及可能的错误。关键点在于:
✅ 你必须直接操作 p 的底层数组;
❌ 不能用 append 创建新切片(这不会影响调用方持有的原切片);
❌ 不应提前返回 1 或固定值——Validate 函数会多次调用 Read 并校验每次是否填满整个缓冲区。
题目要求“emit an infinite stream of 'A'”,本质是让 MyReader 在每次 Read 调用时,把输入缓冲区 b 的每个字节都设为 'A'(即 ASCII 值 65),并返回写入长度。参考实现如下:
package main
import "golang.org/x/tour/reader"
type MyReader struct{}
func (MyReader) Read(b []byte) (int, error) {
for i := range b {
b[i] = 'A'
}
return len(b), nil // 成功写入全部字节,无错误
}
func main() {
reader.Validate(MyReader{}) // 内部会多次调用 Read 并验证结果
}⚠️ 注意事项:
总结:理解 io.Reader 的契约比语法更重要——它是一个消费型接口,你不是“发出”字符,而是“填充”对方提供的容器。掌握这一点,才能真正驾驭 Go 的 I/O 抽象与组合能力。