当前位置:网站首页>用 Antlr 重构脚本解释器
用 Antlr 重构脚本解释器
2022-08-11 07:48:00 【crossoverJie】
前言
在上一个版本实现的脚本解释器 GScript 中实现了基本的四则运算以及 AST
的生成。
当我准备再新增一个 %
取模的运算符时,会发现工作很繁琐而且几乎都是重复的;主要是两步:
- 需要在词法解析器中新增对
%
符号的支持。 - 在语法解析器遍历 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: ("(") 0DECIMAL_LITERAL ("2") 1PLUS ("+") 2DECIMAL_LITERAL ("3") 3 (")") 4MULT ("*") 6DECIMAL_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]?;
完整规则:https://github.com/crossoverJie/gscript/blob/main/GScript.g4
运行:
antlr -Dlanguage=Go -o parser -visitor -no-listener GScript.g4
就可以帮我们生成 Go
的代码(默认是 Java
),关于 Antlr
的词法、文法规则以及安装步骤请参考官网。
而我们要实现具体的语法逻辑时只需要实现相关的接口,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 语法树,可以帮我们更好的调试代码。
升级 xjson
借助 GScript
提供的 statement
,xjson
也提供了有些有意思的写法:
因为 xjson
的四则运算语法没有使用 Antlr
生成,所以为了能支持 GScript
提供的 statement
需要手写许多词法代码。
这也体现了 Antlr
这类前端工具的重要性,效率提升是非常明显的。
总结
借助于 Antlr
后续 GScript
会继续支持函数调用、更完善的类型系统、面向对象等特性;感兴趣的朋友请持续关注。
边栏推荐
- Do you know the basic process and use case design method of interface testing?
- 零基础SQL教程: 主键、外键和索引 04
- 1061 True or False (15 points)
- 我的创作纪念日丨感恩这365天来有你相伴,不忘初心,各自精彩
- 1096 big beautiful numbers (15 points)
- Evolution and New Choice of Streaming Structured Data Computing Language
- oracle19c does not support real-time synchronization parameters, do you guys have any good solutions?
- redis operation
- Pico neo3 Unity Packaging Settings
- [C语言] sscanf如何实现sscanf_s?
猜你喜欢
JRS303-Data Verification
1036 Programming with Obama (15 points)
几何EX3 功夫牛宣布停售,入门级纯电产品为何总成弃子
查找最新人员工资和上上次人员工资的变动情况
零基础SQL教程: 基础查询 05
软件测试常用工具的用途及优缺点比较(详细)
Analysys and the Alliance of Small and Medium Banks jointly released the Hainan Digital Economy Index, so stay tuned!
Project 1 - PM2.5 Forecast
支持各种文件快速重命名最简单的小技巧
1.2 - error sources
随机推荐
string类接口介绍及应用
进阶-指针
2022-08-10 mysql/stonedb-慢SQL-Q16-耗时追踪
美术2.4 UV原理基础
Distributed Lock-Redission - Cache Consistency Solution
C语言-结构体
1061 True or False (15 points)
【C语言】每日一题,求水仙花数,求变种水仙花数
1091 N-Defensive Number (15 points)
1036 跟奥巴马一起编程 (15 分)
tf.cast(),reduce_min(),reduce_max()
关于#sql#的问题:怎么将下面的数据按逗号分隔成多行,以列的形式展示出来
Write a resume like this, easy to get the interviewer
TF中的One-hot
Test cases are hard?Just have a hand
klayout--导出版图为gds文件
The most complete documentation on Excel's implementation of grouped summation
JRS303-数据校验
3.2 - classification - Logistic regression
Active users of mobile banking grew rapidly in June, hitting a half-year high