bufio包通过内存缓冲减少系统调用次数而非加速读写;WriteString后需显式Flush才能写入磁盘;ReadString遇EOF可能报错,须检查err;缓冲区大小应依场景调整;Reader/Writer混用同一文件需避免竞态。
Go 标准库的 bufio 包不是为了“加速”读写,而是为了解决小数据量频繁系统调用带来的开销——直接对 *os.File 每次 Write 或 Read 都可能触发一次系统调用;bufio.Writer 和 bufio.Reader 通过内存缓冲层批量处理,显著降低 syscall 次数。
bufio.Writer 的写入默认只进内存缓冲区,不自动刷到磁盘。常见现象:程序结束但文件为空,或只看到部分数据。
Write / WriteString 只填充内部 buf,未触发实际写入Flush,但不能依赖这个行为w.Flush(),否则缓冲区残留数据丢失
Writer;若确定写完,defer w.Flush() 是安全习惯file, _ := os.Create("out.txt")
w := bufio.NewWriter(file)
w.WriteString("hello")
w.WriteString(" world")
w.Flush() // 必须!否则文件里什么都没有
file.Close()
bufio.Reader.ReadString 会一直读直到遇到指定分隔符(如 '\n'),若输入流提前 EOF 且未找到分隔符,返回 err == io.EOF 或 err == bufio.ErrBufferFull;若忽略错误直接取 string,可能 panic 或读出脏数据。
err,不要假设一定能读到换行符ReadBytes 或 ReadSlice
Scanner 更安全(它内置处理了缓冲和边界),ReadString 更适合协议解析等可控场景bufio.NewReaderSize(r, 16))可能导致单行超长时失败f, _ := os.Open("log.txt")
r := bufio.NewReader(f)
for {
line, err := r.ReadString('\n')
if err == io.EOF {
if len(line) > 0 { // 最后一行可能没换行符
fmt.Println("last:", line)
}
break
}
if err != nil {
log.Fatal(err)
}
fmt.Print("line:", line)
}
默认缓冲区是 4096 字节,适用于大多数文本日志、HTTP 响应等场景。但需根据使用模式调整:
低频大写(如导出一个 10MB CSV):缓冲意义不大,甚至浪费内存;可设为 0(此时退化为直接写底层 io.Writer)bufio.NewWriterSize(w, 0)),避免延迟Flush 变成空操作可以共用一个 *os.File,但必须确保**没有并发读写**,否则文件偏移量错乱、数据覆盖、panic 都可能发生。
Seek 重置位置,否则读写位置不同步sync.Mutex)保护 Reader/Writer 实例,或改用 channel 协调io.Pipe 解耦bufio.NewReader(os.Stdin) 和 fmt.Scanln 混用,后者内部也用缓冲,会互相干扰缓冲的本质是权衡:用内存换 syscall 次数,用可控延迟换吞吐。真正难的不是调用 NewWriter,而是判断什么时候该 flush、什么时候该扩容、以及在并发下谁负责维护文件偏移——这些细节不处理,缓存反而变成 bug 温床。