Go中读写标准I/O最稳妥方式是用bufio包:Scanner适合安全按行读取,Reader适合字节级控制,Writer提升批量写入性能;切忌混用fmt与bufio输入函数,以防缓冲错位。

Go 里读写标准输入输出最常用、最稳妥的方式就是用 bufio 包,而不是直接用 fmt.Scan 或 os.Stdin.Read —— 后者在处理换行、缓冲、多字段输入时容易出错或阻塞。
用 bufio.Scanner 安全读取多行输入
bufio.Scanner 是处理行输入的首选,它自动按行切割、处理缓冲,并能规避 bufio.Reader.ReadString('\n') 可能导致的换行符残留问题。
- 默认每行最大 64KB,超长会报
scanner.ErrTooLong;如需读超长行,得提前调scanner.Buffer设置缓冲区大小 - 遇到 EOF 或 I/O 错误时,
scanner.Scan()返回false,错误需用scanner.Err()显式检查 - 别用
scanner.Text()拿空行后的数据——如果上一次Scan()失败,Text()返回的是上次成功的内容,不是当前状态
scanner := bufio.NewScanner(os.Stdin)
for scanner.Scan() {
line := scanner.Text() // 不含 \n
fmt.Println("got:", line)
}
if err := scanner.Err(); err != nil {
log.Fatal(err)
}
用 bufio.Reader 精确控制字节级输入
当需要读单个字符、跳过空白、或混合读取(比如先读 int 再读一行字符串)时,bufio.Reader 更灵活。
-
reader.ReadString('\n')会包含结尾的\n,记得用strings.TrimSpace或切片去掉 -
reader.ReadRune()适合处理 Unicode 字符(如中文),而ReadByte()只读 ASCII 字节 - 多次调
ReadString或ReadBytes之间不会丢数据,但Scanner和Reader不能混用同一个os.Stdin,否则会相互干扰
reader := bufio.NewReader(os.Stdin)
n, _ := reader.ReadString('\n')
fmt.Printf("raw: %q\n", n) // 比如 "hello\n"
用 bufio.Writer 批量写入提升性能
频繁调 fmt.Print 或 os.Stdout.Write 会产生大量系统调用。bufio.Writer 缓冲后一次性刷出,尤其在循环中写入大量小字符串时效果明显。
立即学习“go语言免费学习笔记(深入)”;
- 必须显式调
writer.Flush(),否则程序退出前可能丢失最后几条输出(特别是没换行的) - 缓冲区大小可自定义:
bufio.NewWriterSize(os.Stdout, 4096),小数据用默认 4KB 就够 - 如果写入内容含大量格式化(如
fmt.Fprintf),优先用fmt.Fprint(w, ...)而非先拼接字符串再w.Write,减少内存分配
w := bufio.NewWriter(os.Stdout) fmt.Fprintln(w, "hello") fmt.Fprint(w, "world") w.Flush() // 必须!
常见陷阱:混用 fmt 和 bufio 导致输入错位
这是新手最容易踩的坑:fmt.Scan、fmt.Scanf 底层直接操作 os.Stdin,和 bufio.Scanner 或 bufio.Reader 共享同一个文件描述符,但缓冲策略不同,会导致“吃掉”本该被 Scanner 读到的输入。
- 例如:先用
fmt.Scan(&n)读一个整数,再用scanner.Scan()读下一行——很可能第二行读到的是空字符串,因为fmt.Scan停在换行符前,而Scanner从换行符开始读,直接跳到下一行 - 统一使用
bufio.Scanner或bufio.Reader,避免和fmt输入函数混用 - 如果必须用
fmt.Scanf解析结构化输入(如"%d %s"),那就全程用它,别切到Scanner
缓冲策略不一致这点,往往要 debug 到 stdin 的字节流层面才能看清,别只盯着逻辑。










