Go不支持内置字符串逻辑表达式求值,需用govaluate等第三方库或手写解析器;govaluate支持优先级、括号及自定义函数,但需显式传参且对nil默认panic。

Go 里没有内置的逻辑表达式解析器,and/or/not 不能直接写成字符串求值
Go 不像 Python 的 eval() 或 JavaScript 的 eval,它在编译期就确定类型和行为,运行时无法动态解析字符串形式的逻辑表达式(比如 "age > 18 and (city == 'Beijing' or not verified)")。你必须自己构造解析逻辑,或借助第三方库。
用 govaluate 解析并计算字符串逻辑表达式最省事
这是目前最稳定、文档清晰、支持括号/优先级/自定义函数的表达式库。它把字符串转成可执行的 AST,再传入变量上下文求值。
- 安装:
go get github.com/Knetic/govaluate - 支持标准操作符:
&&(and)、||(or)、!(not),也兼容and/or/not(需启用govaluate.AllowVariance) - 变量必须显式传入
map[string]interface{},不支持隐式作用域 - 注意:
!只能作用于布尔值或可转为布尔的类型(如非零数字 → true),对 nil 或未定义变量会 panic
package main
import (
"fmt"
"github.com/Knetic/govaluate"
)
func main() {
expr, _ := govaluate.NewEvaluableExpression("age > 18 && (city == 'Beijing' || !verified)")
parameters := map[string]interface{}{
"age": 25,
"city": "Beijing",
"verified": false,
}
result, _ := expr.Evaluate(parameters)
fmt.Println(result) // true
}
手写简单布尔表达式求值器适合可控场景
如果你只需要处理固定结构(如规则引擎中预定义的字段 + 有限操作符),不用支持任意嵌套,可以避开依赖,用递归下降或状态机实现。关键点:
- 先 tokenize:把
"a and b or not c"拆成[]string{"a", "and", "b", "or", "not", "c"} - 注意运算符优先级:
not>and>or,建议转成后缀表达式(逆波兰)再求值 - 变量查表必须明确来源,例如从 struct tag、map 或数据库字段映射,避免运行时 panic
- 不要尝试复用 Go 的语法分析器(
go/parser),它解析的是 Go 源码,不是布尔表达式
真实项目中容易踩的坑
逻辑表达式常用于权限控制、风控规则、配置化策略等场景,但线上出问题往往不是语法错,而是语义失控:
立即学习“go语言免费学习笔记(深入)”;
-
nil值参与and/or时被隐式转成false,但govaluate默认 panic;需提前过滤或用coalesce函数兜底 - 字符串比较默认区分大小写,
"ADMIN" == "admin"是false,业务上可能需要strings.EqualFold—— 必须注册为自定义函数,不能靠表达式本身解决 - 时间比较(如
created_at > '2024-01-01')需要先将字符串 parse 成time.Time,否则只是字典序比较 - 表达式超时或无限循环(比如递归过深、正则回溯)没做限制,可能拖垮服务 —— 建议加 context 和最大深度/步骤数限制
真正难的从来不是怎么算 a && b || !c,而是怎么让运营人员写的表达式,在上线前被校验、有类型提示、失败时给出可读错误,且不因一个错别字导致整批请求 500。










