fmt.Scan和fmt.Scanln是Go中基础输入函数:Scan按空白分隔读取任意数量值,不关心换行;Scanln仅读当前行且要求参数与输入值数量严格匹配,适合逐行输入场景。

在 Go 语言中,fmt.Scan 和 fmt.Scanln 是最基础的从标准输入读取数据的方式,适合简单命令行交互场景。它们会自动解析输入内容并赋值给变量,但使用时需注意类型匹配、换行处理和错误忽略等细节。
Scan:读取空格/制表符/换行分隔的任意数量值
fmt.Scan 按空白字符(空格、tab、换行)切分输入,直到填满所有参数变量为止,不关心换行位置。它会跳过开头的空白,并在遇到第一个非空白后开始读取,直到下一个空白或 EOF。
示例:
输入一行:123 hello true
var a int var b string var c bool fmt.Scan(&a, &b, &c) // a=123, b="hello", c=true
- 如果输入是
123\nhello\ntrue(每项换一行),Scan同样能正确读取 - 若只提供两个变量却输入三个值,第三个值会留在缓冲区,影响后续 Scan 调用
- 不会因换行而中断读取,适合“一行多值”或“多行连续输入”的宽松场景
Scanln:只读取单行,且要求输入以换行结束
fmt.Scanln 只读取**当前行**的内容,遇到换行就停止。它要求输入的值个数必须与参数个数严格一致,否则返回 err != nil(常见为 fmt.ErrTooManyValues 或 fmt.ErrUnexpectedEOF)。
立即学习“go语言免费学习笔记(深入)”;
示例:
输入:456 world false
var x int var y string var z bool _, err := fmt.Scanln(&x, &y, &z) // 成功
- 输入
789 test却声明了三个变量 → 报错:err = fmt.ErrUnexpectedEOF - 输入
100 ok yes extra→ 报错:err = fmt.ErrTooManyValues - 适合需要“严格一行一完整记录”的交互,比如让用户逐行输入配置项
实用建议与常见陷阱
两者都返回 n int, err error,但很多人直接忽略 err,导致输入异常时程序静默失败。
- 始终检查
err,尤其在教学、CLI 工具或用户输入不可控时 - 想读字符串含空格?
Scan和Scanln都不行 —— 改用bufio.NewReader(os.Stdin).ReadString('\n')或fmt.Scanln不适用,应选bufio.ReadString - 多次调用前,注意缓冲区残留:例如
Scanln读完一行后,下一次Scan可能立刻读到上一行剩下的值(如果之前没清空) - 数字输入错误(如输字母)会导致
err且变量保持零值,可结合fmt.Sscanf或自定义解析增强健壮性
一个安全读取整数的小例子
var num int
for {
fmt.Print("请输入一个整数: ")
_, err := fmt.Scanln(&num)
if err == nil {
fmt.Printf("你输入的是: %d\n", num)
break
}
fmt.Println("输入无效,请重试。")
// 清空输入缓冲区(防止死循环)
bufio.NewReader(os.Stdin).ReadBytes('\n')
}
基本上就这些。Scan 和 Scanln 简单直接,适合入门和轻量交互;真要处理复杂输入,建议搭配 bufio 或结构化解析库更稳妥。










