应使用 strconv.ParseFloat 解析用户输入的浮点数并校验 error,配合 strings.TrimSpace 去除空白;四则运算需封装为返回 (float64, error) 的函数,显式检查除零、NaN、Inf;用 switch 处理运算符,bufio.Scanner 替代 fmt.Scanln 保证交互健壮。

用 strconv.ParseFloat 解析用户输入的数字,别直接用 fmt.Scanf 读浮点数
Go 的 fmt.Scanf 对浮点数输入容错极差,遇到空格、换行或非法字符会静默失败或阻塞。真实场景中用户可能输 "12.5 " 或 "-3.14abc",必须显式校验。
- 始终用
strconv.ParseFloat(input, 64)转换,并检查返回的error - 用
strings.TrimSpace去首尾空白,避免" 42 "导致解析失败 - 设置合理精度(如
64)——32在计算0.1 + 0.2时误差更大
把四则运算封装成带错误返回的函数,比如 func add(a, b float64) (float64, error)
看似简单的加减乘除,在计算器里必须考虑边界:除零、溢出、NaN 传播。Go 不支持异常,所有异常路径都得走 error 返回。
-
divide函数必须显式判断b == 0,返回自定义错误如errors.New("division by zero") - 乘法和加法虽不直接 panic,但若输入是
+Inf或NaN,结果不可控,建议在运算前用math.IsNaN和math.IsInf过滤 - 不要在函数内
log.Fatal或panic——调用方应决定如何处理错误(重试?提示?退出?)
func divide(a, b float64) (float64, error) {
if b == 0 {
return 0, errors.New("division by zero")
}
result := a / b
if math.IsNaN(result) || math.IsInf(result, 0) {
return 0, errors.New("invalid arithmetic result")
}
return result, nil
}
用 switch 处理运算符字符串,别写一堆 if/else if
用户输入的运算符是字符串(如 "+", "*"),用 switch 不仅可读性高,还能天然防大小写混淆和多余空格问题。
- 先用
strings.TrimSpace清理运算符输入,再switch op - 每个
case调用对应运算函数并检查错误,不要把错误处理逻辑分散到各处 - 必须有
default分支返回明确错误,比如"unsupported operator: " + op,否则非法符号会静默失败
主循环里用 bufio.Scanner 替代 fmt.Scanln,避免换行残留干扰
fmt.Scanln 读完一行后,如果输入缓冲区还有未消费的换行符,下一次读取可能立即返回空字符串。交互式计算器连续输入时极易卡死或跳过输入。
立即学习“go语言免费学习笔记(深入)”;
- 用
bufio.NewScanner(os.Stdin),它按行读取且自动丢弃换行符 - 每次调用
scanner.Scan()后,用scanner.Text()获取干净字符串 - 注意检查
scanner.Err()——例如用户按 Ctrl+D(EOF)时需优雅退出,而不是 panic
真正难的不是算加减法,是让程序在用户随手乱输、反复回退、中间 Ctrl+C 的情况下,依然能给出明确反馈、不崩溃、不卡住。每一步输入解析和每一步运算都要假设它会失败——Go 的错误处理不是装饰,是运行时的呼吸节奏。










