在解释型语言中基本都提供了 eval() 函数,将字符串当做代码来执行。而golang并没有提供此方法。我在编写计算器桌面应用的时候发现,必须要有 eval 函数才能实现。
PaulXu-cn/goeval
https://github.com/PaulXu-cn/goeval
在临时目录os.TempDir()
下创建随机目录,然后创建main.go
文件,填充go源码,运行完然后将其删掉,获取stdout信息。
// github.com/PaulXu-cn/goeval
func Eval(defineCode string, code string, imports ...string) (re []byte, err error)
第一个参数填入自定义的变量/函数/结构体等。
第二个参数是代码字符串。
第三个参数是本代码依赖了哪些包,可以是标准库,也可以是第三方包。
func t11() {
res, err := goeval.Eval("", "fmt.Println(3*(1+2))", "fmt")
fmt.Println(string(res), err)
}
func t12() {
res, err := goeval.Eval("", `line := "3*(1+1)"
expr, _ := govaluate.NewEvaluableExpression(line)
result, _ := expr.Evaluate(nil)
fmt.Println(result)`, "fmt", "github.com/Knetic/govaluate")
fmt.Println(string(res), err)
}
代码格式化:因为Go有些地方对format有要求,比如定义函数的时候左大括号不能换行,使用go/format
来进行格式化。
go.mod 的处理:如果引入了标准库以外的包,就需要初始化go.mod文件,但是PaulXu-cn/goeval
没有这个操作,代码也能正常运行,主要是因为,这是在当前项目下运行一个tmp项目下的main.go
,于是go命令会检查当前项目下的go.mod,而不是tmp项目下的go.mod,而在当前项目下我实际上也在使用github.com/Knetic/govaluate
包,这就导致可以正常运行,但是,如果 eval 代码串中引入的包,在当前项目下并没有用到,就会报错,这个时候你可以使用下划线语法将其引入。
xtaci/goeval
https://github.com/xtaci/goeval
此项目式借助 go/ast, go/parser
解析抽象语法树来实现的。
func t2() {
s := evalhandle.NewScope()
s.Set("print", fmt.Println)
fmt.Println(s.Eval(`count := 0`))
fmt.Println(s.Eval(`for i:=0; i<10; i++ {
count=count+i
}`))
fmt.Println(s.Eval(`print(count)`))
}
Knetic/govaluate
https://github.com/Knetic/govaluate
计算一个表达式的值并得到返回结果。
func t3() {
line := "3*(1+2)"
expr, _ := govaluate.NewEvaluableExpression(line)
result, _ := expr.Evaluate(nil)
fmt.Println(result) // 9
}
最终使用了govaluate
完成了计算器的开发。