应使用 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")
+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 读完一行后,如果输入缓冲区还有未消费的换行符,下一次读取可能立即返回空字符串。交互式计算器连续输入时极易卡死或跳过输入。
bufio.NewScanner(os.Stdin),它按行读取且自动丢弃换行符scanner.Scan() 后,用 scanner.Text() 获取干净字符串scanner.Err() ——例如用户按 Ctrl+D(EOF)时需优雅退出,而不是 panic
真正难的不是算加减法,是让程序在用户随手乱输、反复回退、中间 Ctrl+C 的情况下,依然能给出明确反馈、不崩溃、不卡住。每一步输入解析和每一步运算都要假设它会失败——Go 的错误处理不是装饰,是运行时的呼吸节奏。