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

用 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 变成一句人话反馈——这部分没标准解法,得自己一层层拦。










