一个基于ANTLR 4的布尔表达式语句解释器的实现
Reference
The Definitive ANTLR 4 Reference, 2nd Edition.
0 Features
labeled grammar definition, i.e. # eq;
Visitorpattern.
1 The Grammar
grammar RuleSet;
prog: stmt+ ;
stmt: expr '=' expr NEWLINE #eq
|expr '<' expr NEWLINE #lt
|expr '<=' expr NEWLINE # le
|expr '>' expr NEWLINE # gt
|expr '>=' expr NEWLINE # ge
;
expr: expr op=('*'|'/') expr # MulDiv
| expr op=('+'|'-') expr # AddSub
| INT # int
| '(' expr ')' # parens
;
MUL : '*' ; // assigns token name to '*' used above in grammar
DIV : '/' ;
ADD : '+' ;
SUB : '-' ;
NEWLINE : [\r\n]+ ;
INT : [0-9]+ ; // match integers
WS : [ \t]+ -> skip ; // toss out whitespace
2 Play around with ANTLR 4
export CLASSPATH=".:./antlr-4.5.1-complete.jar:$CLASSPATH"
# generate listner
#java -jar ./antlr-4.5.1-complete.jar RuleSet.g4
# generate visitor
java -jar ./antlr-4.5.1-complete.jar -no-listener -visitor RuleSet.g4
3 The Interpreter
Put the generate java files in pakage com.spike.antlr4.rs.
3.1 The Visitor
package com.spike.antlr4.rs;
import com.spike.antlr4.rs.RuleSetParser.ExprContext;
public class EvalVisitor extends RuleSetBaseVisitor<Integer> {
private boolean result = false;
public boolean getResult() {
return result;
}
public void setResult(boolean result) {
this.result = result;
}
/** = */
@Override
public Integer visitEq(RuleSetParser.EqContext ctx) {
// Integer result = visitChildren(ctx);
ExprContext left = ctx.expr(0);
ExprContext right = ctx.expr(1);
int result = visit(left) - visit(right);
if (result == 0) {
this.setResult(true);
} else {
this.setResult(false);
}
return result;
}
/** <= */
@Override
public Integer visitLe(RuleSetParser.LeContext ctx) {
// Integer result = visitChildren(ctx);
ExprContext left = ctx.expr(0);
ExprContext right = ctx.expr(1);
int result = visit(left) - visit(right);
if (result <= 0) {
this.setResult(true);
} else {
this.setResult(false);
}
return result;
}
/** < */
@Override
public Integer visitLt(RuleSetParser.LtContext ctx) {
ExprContext left = ctx.expr(0);
ExprContext right = ctx.expr(1);
int result = visit(left) - visit(right);
if (result < 0) {
this.setResult(true);
} else {
this.setResult(false);
}
return result;
}
/** >= */
@Override
public Integer visitGe(RuleSetParser.GeContext ctx) {
ExprContext left = ctx.expr(0);
ExprContext right = ctx.expr(1);
int result = visit(left) - visit(right);
if (result >= 0) {
this.setResult(true);
} else {
this.setResult(false);
}
return result;
}
/** > */
@Override
public Integer visitGt(RuleSetParser.GtContext ctx) {
ExprContext left = ctx.expr(0);
ExprContext right = ctx.expr(1);
int result = visit(left) - visit(right);
if (result > 0) {
this.setResult(true);
} else {
this.setResult(false);
}
return result;
}
/** INT */
@Override
public Integer visitInt(RuleSetParser.IntContext ctx) {
return Integer.valueOf(ctx.INT().getText());
}
/** expr op=('*'|'/') expr */
@Override
public Integer visitMulDiv(RuleSetParser.MulDivContext ctx) {
int left = visit(ctx.expr(0)); // get value of left subexpression
int right = visit(ctx.expr(1)); // get value of right subexpression
if (ctx.op.getType() == RuleSetParser.MUL)
return left * right;
return left / right; // must be DIV
}
/** expr op=('+'|'-') expr */
@Override
public Integer visitAddSub(RuleSetParser.AddSubContext ctx) {
int left = visit(ctx.expr(0)); // get value of left subexpression
int right = visit(ctx.expr(1)); // get value of right subexpression
if (ctx.op.getType() == RuleSetParser.ADD)
return left + right;
return left - right; // must be SUB
}
/** '(' expr ')' */
@Override
public Integer visitParens(RuleSetParser.ParensContext ctx) {
return visit(ctx.expr()); // return child expr's value
}
}
3.2 The Main Entrance
package com.spike.antlr4.rs;
import java.util.ArrayList;
import java.util.List;
import org.antlr.v4.runtime.ANTLRInputStream;
import org.antlr.v4.runtime.CommonTokenStream;
import org.antlr.v4.runtime.tree.ParseTree;
/**
* The Interpreter
*
* @author zhoujiagen<br/>
* Aug 26, 2015 1:20:34 AM
*/
public class Calc {
static String NEWLINE = "\r\n";
public static void main(String[] args) {
System.out.println(eval("123=100+12+11" + NEWLINE, "124 + 125 = 123 + 126" + NEWLINE, "345 <= 346 + 347 + 348"
+ NEWLINE));
}
static List<Boolean> eval(String... stmts) {
if (stmts != null && stmts.length > 0) {
List<Boolean> result = new ArrayList<Boolean>();
for (String stmt : stmts) {
result.add(eval(stmt));
}
return result;
}
return null;
}
static boolean eval(String stmt) {
boolean result = false;
if (stmt != null && stmt.length() > 0) {
ANTLRInputStream input = new ANTLRInputStream(stmt);
RuleSetLexer lexer = new RuleSetLexer(input);
CommonTokenStream tokens = new CommonTokenStream(lexer);
RuleSetParser parser = new RuleSetParser(tokens);
ParseTree tree = parser.stmt(); // parse
EvalVisitor eval = new EvalVisitor();
System.out.println(eval.visit(tree));
// get the final result
result = eval.getResult();
}
return result;
}
}
一个基于ANTLR 4的布尔表达式语句解释器的实现的更多相关文章
- 初识NodeJS,一个基于GoogleV8引擎的Javascript运行环境
思考 首先我们来思考一个问题:我们都知道几乎所有现代主流浏览器都全面支持了ECMAScript 5.1版标准,而JavaScript的标准是ECMAScript.那么我们就容易认为JavaScript ...
- 一个基于.NET平台的自动化/压力测试系统设计简述
AutoTest系统设计概述 AutoTest是一个基于.NET平台实现的自动化/压力测试的系统,可独立运行于windows平台下,支持分布式部署,不需要其他配置或编译器的支持.(本质是一个基于协议的 ...
- 构建一个基于 Spring 的 RESTful Web Service
本文详细介绍了基于Spring创建一个“hello world” RESTful web service工程的步骤. 目标 构建一个service,接收如下HTTP GET请求: http://loc ...
- JAVA WEB快速入门之从编写一个基于SpringBoot+Mybatis快速创建的REST API项目了解SpringBoot、SpringMVC REST API、Mybatis等相关知识
JAVA WEB快速入门系列之前的相关文章如下:(文章全部本人[梦在旅途原创],文中内容可能部份图片.代码参照网上资源) 第一篇:JAVA WEB快速入门之环境搭建 第二篇:JAVA WEB快速入门之 ...
- 一个基于ASP.NET(C#)的ACCESS数据库操作类
using System; using System.Collections; using System.Collections.Specialized; using System.Data; usi ...
- 一个基于NodeJS开发的APP管理CMS系统
花了大概3周独立开发了一个基于NodeJS的CMS系统,用于公司APP的内容管理( **公司APP?广告放在最后 ^_^ ** ,管理员请理解~~~ )晚上看了部电影还不想睡,闲着也是闲着就作下小小总 ...
- Go/Python/Erlang编程语言对比分析及示例 基于RabbitMQ.Client组件实现RabbitMQ可复用的 ConnectionPool(连接池) 封装一个基于NLog+NLog.Mongo的日志记录工具类LogUtil 分享基于MemoryCache(内存缓存)的缓存工具类,C# B/S 、C/S项目均可以使用!
Go/Python/Erlang编程语言对比分析及示例 本文主要是介绍Go,从语言对比分析的角度切入.之所以选择与Python.Erlang对比,是因为做为高级语言,它们语言特性上有较大的相似性, ...
- Stateless是一个基于C#创建状态机的简单库
Stateless是一个基于C#创建状态机的简单库 .Net轻量状态机Stateless 很多业务系统开发中,不可避免的会出现状态变化,通常采用的情形可能是使用工作流去完成,但是对于简单场景下,用工作 ...
- SDKMAN一个基于命令行界面的SDK用户环境管理程序
1.背景 使用过Python开发的朋友,应该了解到Python2和Python3语法的差异,有时候从网上下载了基于不同解释器的代码,要来回切换版本, 使用起来不是很方便,有时候甚至很麻烦.于是有人发明 ...
随机推荐
- python __file__ 与argv[0]
在python下,获取当前执行主脚本的方法有两个:sys.argv[0]和__file__. sys.argv[0] 获取主执行文件路径的最佳方法是用sys.argv[0],它可能是一个相对路径,所以 ...
- css+div打造三角形(箭头)
在很多网站都见过这样的箭头,之前我一直以为是图片,直到今天才知道原来可以用css做.开始看代码没太看懂,后来自己试了几遍才恍然大悟.贴出来分享下.(大神请直接忽略) 先看代码: HTML部分就是一个单 ...
- velocity常用语句速查表
velocity常用语句 * 变量定义 #set($directoryRoot = "www" ) * #if($!list.size() != 0) //判断list不为空 #f ...
- MFC编程入门之十一(对话框:模态对话框及其弹出过程)
加法计算器对话框程序大家照着做一遍后,相信对基于对话框的程序有了些解了,有个好的开始对于以后的学习大有裨益.趁热打铁,这一节讲讲什么是对话框和非模态对话框,以及模态对话框怎样弹出. 一.模态对话框和非 ...
- Decorator
1 意图:动态地给一个对象添加一些额外的职责.就增加功能来说,Decorator模式相比生成子类更灵活. 2 别名:包装器Wrapper 3 动机:将组件嵌入到另一个对象中,由这个对象添加边框.嵌入的 ...
- SurfaceOutput
下面着重分析一下surface shader中是如何使用BlinnPhong光照模型的,为以后写代码做铺垫.源文件取自Unity3D官网下载的builtin_shaders, 源文件名为Normal- ...
- canvas初体验之基本线条
有的时候我们打开一些网站,可以看到背景是闪烁的星空或者是有一些可以与鼠标交互的线条等等,此酷炫的效果就是用到了html5的canvas效果. 首先来认识一下h5新增的标签的写法<canvas&g ...
- @keyframes
通过 @keyframes 规则,您能够创建动画. @keyframes movelike{ from{right:1205px;} to{right:0px}} 创建动画的原理是,将一套 CSS 样 ...
- Windows Store App 偏移特效
通过前面讲解的内容,读者已经了解了如何在三维空间中使旋转对象绕指定的旋转中心旋转一定的角度.接下来在这个基础上进一步讲解如何对旋转对象进行平移.下面首先介绍一下用到的几个属性. q LocalOff ...
- F2工作流引擎之 工作流运转模型(三)
1流程单起点单终止模型 单起点:一个流程定义必须有且唯一起点 单结束点:一个流程定义必须有且唯一结束点. 约定:提单与结束是每个流程必须有的活动,且唯一只有一个提单和结束. 2串行模型 描述:串行(S ...