go Antlr重构脚本解释器实现示例

作者:crossoverJie 时间:2024-04-26 17:19:44 

前言

在上一个版本实现的脚本解释器 GScript 中实现了基本的四则运算以及 AST 的生成。

go Antlr重构脚本解释器实现示例

当我准备再新增一个 取模的运算符时,会发现工作很繁琐而且几乎都是重复的;主要是两步:

  • 需要在词法解析器中新增对 符号的支持。

  • 在语法解析器遍历 AST 时对 token 实现具体逻辑。

其中的词法解析和遍历 AST 完全是重复工作,所以我们可否能够简化这两步呢?

Antlr

Antlr 就是做帮我们解决这些问题的常用工具,利用它我们只需要编写词法文件,然后就可以自动生成词法、语法解析器,并且可以生成不同语言的代码。

下面以 GScript 的示例来看看 antlr 是如何帮我们生成词法分析器的。

func TestGScriptVisitor_Visit_Lexer(t *testing.T) {
expression := "(2+3) * 2"
input := antlr.NewInputStream(expression)
lexer := parser.NewGScriptLexer(input)
for {
t := lexer.NextToken()
if t.GetTokenType() == antlr.TokenEOF {
break
}
fmt.Printf("%s (%q) %d\n",
lexer.SymbolicNames[t.GetTokenType()], t.GetText(),t.GetColumn())
}
}
//output:
("(") 0
DECIMAL_LITERAL ("2") 1
PLUS ("+") 2
DECIMAL_LITERAL ("3") 3
(")") 4
MULT ("*") 6
DECIMAL_LITERAL ("2") 8

Antlr 会自动将我们的表达式解析为 token,遍历 token 时还能拿到该 token 所在的代码行数、位置等信息,在编译期间做语法检查非常有用。

要实现这些我们只需要编写词法、语法规则文件即可。

刚才的示例所对应的词法、语法规则如下:

expr
   : '(' expr ')'                        #NestedExpr
   | liter=literal #Liter
   | lhs=expr bop=( MULT | DIV ) rhs=expr #MultDivExpr
   | lhs=expr bop=MOD rhs=expr            #ModExpr
   | lhs=expr bop=( PLUS | SUB ) rhs=expr #PlusSubExpr
   | expr bop=(LE | GE | GT | LT ) expr # GLe
   | expr bop=(EQUAL | NOTEQUAL) expr # EqualOrNot
   ;
DECIMAL_LITERAL:    ('0' | [1-9] (Digits? | '_'+ Digits)) [lL]?;    

完整规则:github.com/crossoverJi…

运行:

antlr -Dlanguage=Go -o parser -visitor -no-listener GScript.g4

而我们要实现具体的语法逻辑时只需要实现相关的接口,Antlr 会自动遍历 AST(当然也可以手动控制),同时在访问不同的 AST 节点时会回调我们自己实现的接口,这样我们就能编写自己的语法规则了。

以这里的新增的取模运算为例:

func (v *GScriptVisitor) VisitModExpr(ctx *parser.ModExprContext) interface{} {
lhs := v.Visit(ctx.GetLhs())
rhs := v.Visit(ctx.GetRhs())
return lhs.(int) % rhs.(int)
}

Antlr 回调 VisitModExpr 方法时,便能获取到 % 符号左右两侧的数据,这时只需要做相关运算即可。

基于这个模式这次新增了一个 statement,具体语法如下:

func TestGScriptVisitor_VisitIfElse8(t *testing.T) {
expression := `
if(3!=(1+2)){
return 1+3
} else {
return false
}`
input := antlr.NewInputStream(expression)
lexer := parser.NewGScriptLexer(input)
stream := antlr.NewCommonTokenStream(lexer, 0)
parser := parser.NewGScriptParser(stream)
parser.BuildParseTrees = true
tree := parser.Prog()
visitor := GScriptVisitor{}
var result = visitor.Visit(tree)
fmt.Println(expression, " result:", result)
assert.Equal(t, result, false)
}

Antlr 还有其他各种优势,比如可以解决:

  • 左递归。

  • 二义性。

  • 优先级。

等问题。

这里也推荐在 IDE 中安装 Antlr 的插件,这样就可以直观的查看 AST 语法树,可以帮我们更好的调试代码。

go Antlr重构脚本解释器实现示例

go Antlr重构脚本解释器实现示例

升级 xjson

借助 GScript 提供的 statementxjson 也提供了有些有意思的写法:

go Antlr重构脚本解释器实现示例

因为 xjson 的四则运算语法没有使用 Antlr 生成,所以为了能支持 GScript 提供的 statement 需要手写许多词法代码。

go Antlr重构脚本解释器实现示例

这也体现了 Antlr 这类前端工具的重要性,效率提升是非常明显的。

来源:https://juejin.cn/post/7129357242159071240

标签:go,Antlr,脚本,解释器
0
投稿

猜你喜欢

  • 关于mysql主备切换canal出现的问题解决

    2024-01-28 14:29:08
  • Pyinstaller 打包exe教程及问题解决

    2023-01-08 01:40:00
  • Python字典的核心底层原理讲解

    2022-03-26 08:31:09
  • Extjs显示从数据库取出时间转换JSON后的出现问题

    2024-01-18 12:52:13
  • Python中的id()函数指的什么

    2022-01-14 12:38:04
  • sql2000挂起无法安装的问题的解决方法

    2024-01-20 19:09:34
  • python-for x in range的用法(注意要点、细节)

    2022-11-12 22:40:49
  • PowerDesigner16生成SQL2005列注释的方法

    2024-01-26 06:06:36
  • javascript中var与let、const的区别详解

    2024-05-09 15:07:32
  • 利用Python实现Windows下的鼠标键盘模拟的实例代码

    2023-06-22 04:37:31
  • flask循环导入的问题解决

    2023-10-01 10:28:52
  • python logging模块的使用总结

    2021-11-15 06:46:53
  • mysql中数据库覆盖导入的几种方式总结

    2024-01-19 22:26:33
  • ASP中数据库调用中常见错误的现象和解决

    2007-09-20 13:24:00
  • Python 机器学习之线性回归详解分析

    2023-08-12 09:48:13
  • Golang函数这些神操作你知道哪些

    2024-04-26 17:22:15
  • python计算机视觉实现全景图像拼接示例

    2021-02-12 09:48:05
  • 基于Python函数和变量名解析

    2022-10-26 13:49:56
  • python logging 日志的级别调整方式

    2021-09-15 10:08:29
  • SQL Server提示"选定的用户拥有对象,所以无法除去该用户”

    2024-01-22 03:35:05
  • asp之家 网络编程 m.aspxhome.com