如何使用Golang实现一个简单的命令行计算器_处理算术表达式

4次阅读

go/parser解析算术表达式需包装为完整语句(如”_ = 1 + 2″),手动提取ast.binaryexpr等节点,不支持变量和函数调用,错误提示不友好;手写50行递归下降解析器更简单可控,需自定义分词、处理负号歧义、统一float64计算并智能格式化输出。

如何使用Golang实现一个简单的命令行计算器_处理算术表达式

go/parser 解析算术表达式最直接但有坑

Go 标准库不提供轻量级表达式求值器,go/parser 是少数能真正解析合法 Go 算术语法(如 1 + 2 * 3)的方案,但它默认解析的是 Go 语句,不是纯表达式——直接传 "1+2" 会报 syntax Error: unexpected EOF

  • 必须包装成完整表达式语句:"_ = 1 + 2""func() { _ = 1 + 2 }"
  • 解析后要手动提取 ast.BinaryExpr / ast.ParenExpr 等节点,忽略 ast.AssignStmt 外壳
  • 不支持变量、函数调用(如 sqrt(4)),只认字面量和基础运算符 + - * / %
  • 错误提示是 Go 编译器风格(如 syntax error: unexpected semicolon),对用户不友好

手写递归下降解析器比想象中简单且可控

如果你只需要四则运算 + 括号,写一个 50 行左右的递归下降解析器,比硬啃 go/parser 更快、更易调试、更容易加错误定位。

  • 优先级靠函数嵌套实现:ParseExpr()ParseTerm()ParseTerm()ParseFactor()
  • 每个函数消耗 Token 流(比如用 []rune 或自定义 lexer),遇到非法字符立刻返回带位置的错误
  • 支持浮点数:1.5 + 2e-1 可以用 strconv.ParseFloat 处理,别用 int 硬转
  • 空格和制表符必须跳过,否则 "1 +2" 会卡在 + 后的空格上

shlex 类库不能直接用于算术表达式分词

有人想用类似 Python 的 shlex 库切分表达式,比如把 "1 + (2 * 3)" 拆成 ["1", "+", "(", "2", "*", "3", ")"],但 Go 没有标准 shlex,第三方包(如 github.com/google/shlex)专为 shell 命令设计,会把 -1 当作选项、把 1.5 拆成 ["1", ".", "5"],完全不可用。

  • 算术表达式分词必须自己写:按字符扫描,累积数字(含小数点和 e)、识别运算符、跳过空白
  • 负号 - 是难点:它可能是减号(二元),也可能是负号(一元),需结合上下文判断(比如开头、左括号后、运算符后)
  • 别用 strings.Fields —— 它按空白切,会把 1+2 当成一个 token

输出浮点结果时小心整数除法陷阱

用户输入 5 / 2,期望得到 2.5,但如果内部用 int 运算,结果就是 2;如果全用 float64,又可能让 1 + 2 输出成 3.0,显得啰嗦。

立即学习go语言免费学习笔记(深入)”;

  • 统一用 float64 计算,避免溢出和精度丢失(比如大整数相乘)
  • 显示时判断是否为整数:if math.Floor(v) == v { fmt.printf("%.0f", v) },否则用 %.6g
  • 注意 0.1 + 0.2 != 0.3,但命令行计算器用户通常不关心这个层级的误差,不用上 decimal 包
  • 除零要显式检查,panic 或返回错误,别等 Inf 冒出来再处理

真正麻烦的从来不是加减乘除,而是怎么把用户敲错的 1++2、漏括号的 (1 + 2 * 3、或者中文符号 1+2 变成一句人话反馈——这部分没标准解法,得自己一层层拦。

text=ZqhQzanResources