fmt.Scan和fmt.Scanln是Go中基础输入函数:Scan按空白分隔读取任意数量值,不关心换行;Scanln仅读当前行且要求参数与输入值数量严格匹配,适合逐行输入场景。
在 Go 语言中,fmt.Scan 和 fmt.Scanln 是最基础的从标准输入读取数据的方式,适合简单命令行交互场景。它们会自动解析输入内容并赋值给变量,但使用时需注意类型匹配、换行处理和错误忽略等细节。
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 同样能正确读取fmt.Scanln 只读取**当前行**的内容,遇到换行就停止。它要求输入的值个数必须与参数个数严格一致,否则返回 err != nil(常见为 fmt.ErrTooManyValues 或 fmt.ErrUnexpectedEOF)。
示例:
输入: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 或结构化解析库更稳妥。