java编写简单的语法分析预测程序
编译原理课程中,编了一个简单的语法分析预测程序,这个程序时根据固定的文法得到预测分析表,然后编写程序来判断表达式是否会正确推到出来。
前提是程序没有左递归符合LL(1)文法:
文法如下:
E→TE'
E’ →+TE'|ε
T→FT'
T’ →*FT'|ε
F→(E)|i
为了程序便于编写将E'替换为e,T'替换为t
(2)FIRST集
FIRST(E)={(,i};
FIRST(E’)={+, ε};
FIRST(T)={(,i};
FIRST(T’)={ *, ε};
FIRST(F)={(,i};
(3)FALLOW集
FOLLOW(E)={),#};
FOLLOW(E’)={),#};
FOLLOW(T)={+,),#};
FOLLOW(T’)={+,),#};
FOLLOW(F)={*,+,),#};
(4)预测分析表
|
i |
+ |
* |
( |
) |
# |
|
|
E |
E->TE’ |
E->TE’ |
||||
|
E’ |
E’->+TE’ |
E’->ε |
E’->ε |
|||
|
T |
T->FT’ |
T->FT’ |
||||
|
T’ |
T’->ε |
T’->*FT’ |
T’->ε |
T’->ε |
||
|
F |
F->i |
F->(E) |
一、Stack.java
public class Stack {
private char s[];
private int top;
private int base;
private final int MAX = 200;
/**
*
* <p>Title: </p>
* <p>Description: 初始化栈</p>
*/
public Stack() {
initStack();
}
private void initStack() {
s = new char[MAX];
base = 0;
top = 0;
}
public boolean isEmpty() {
if(top == base) {
return true;
}else {
return false;
}
}
/**
*
* <p>Title: getTop</p>
* <p>Description: 获取栈顶元素</p>
* @return 返回栈顶元素
*/
public char getTop() {
return s[top-1];
}
/**
*
* <p>Title: push</p>
* <p>Description: 进栈方法</p>
* @param str 进栈的字符
*/
public void push(String str) {
for (int i = str.length() - 1; i >= 0; i--) {
s[top++] = str.charAt(i);
}
}
/**
*
* <p>Title: clear</p>
* <p>Description: 清空栈</p>
*/
public void clear() {
top = base;
}
/**
*
* <p>Title: pop</p>
* <p>Description: 出栈</p>
* @return 栈顶元素出栈并返回出栈的元素
*/
public char pop() {
return s[--top];
}
/**
*
* <p>Title: size</p>
* <p>Description: 返回栈中元素个数</p>
* @return 栈中元素个数
*/
public int size() {
return top;
}
/**
* 打印栈里面的元素
*/
public String toString() {
StringBuffer tempStack = new StringBuffer();
for (int i = 0; i < top; i++) {
tempStack.append(s[i]);
}
return tempStack.toString();
}
}
二、GrammarAnalyze.java
package grammarAnalyze;
public class GrapparAnalyze {
//分析表将E'替换为e,T'替换t
private String tab[][] = {
{ "$", "i", "+", "*", "(", ")", "#" },
{ "E", "Te", "$", "$", "Te", "$", "$" },
{ "e", "$", "+Te", "$", "$", "ε", "ε" },
{ "T", "Ft", "$", "$", "Ft","$", "$" },
{ "t", "$", "ε", "*Ft", "$", "ε", "ε" },
{ "F", "i", "$", "$", "(E)","$", "$" } };
private String input; //input中存放输入的表达式
private StringBuffer tempBuffer; //存放要输出的字符串
private int ptr, row, col, step; //指针,预测表中的行,列,当前步骤
private boolean symbol;
private Stack stack;
public GrapparAnalyze(){
stack = new Stack();
tempBuffer = new StringBuffer();
symbol=true;
input="";
stack.clear();
stack.push("#");
row=1;
ptr=0;
step=1;
}
public int column(char c) { //判断预测表中的列
switch (c) {
case 'i':
return 1;
case '+':
return 2;
case '*':
return 3;
case '(':
return 4;
case ')':
return 5;
case '#':
return 6;
default:
return -1;
}
}
public int line(char c) { //判定预测表中的行
switch (c) {
case 'E':
return 1;
case 'e':
return 2;
case 'T':
return 3;
case 't':
return 4;
case 'F':
return 5;
default:
return -1;
}
}
public void show(String str) {
tempBuffer.append(String.format("%-7d%-14s%-20s%-20s\r\n",
step, filter(stack.toString()), filter(input.substring(ptr)), filter(str)));
step++;
}
public void analyse() {
stack.push(tab[row][0]); //预测表中的第一个元素‘E’
show("初始化");
String tmp;
char ctmp; //栈顶元素
while (!(input.charAt(ptr) == '#' && stack.getTop() == '#')) {
ctmp = stack.getTop();//获取栈顶的元素
if (input.charAt(ptr) == ctmp) { //与栈顶元素比较
stack.pop();
ptr++;
show("" + ctmp + "匹配");
continue;
}
//判断ptr位置的终结符所在预测表的列位置
col = column(input.charAt(ptr));
if (col == -1) {
symbol = false;
show("未定义的字符");
ptr++;
break;
}
//判断栈顶位置所在预测表的行位置
row = line(ctmp);
if (row == -1) {
symbol = false;
show("出错");
stack.pop();
if (input.charAt(ptr) != '#') {
ptr++;
}
continue;
}
tmp = tab[row][col];
if (tmp.charAt(0) == '$') {
symbol = false;
show("出错");
stack.pop();
if (input.charAt(ptr) != '#') {
ptr++;
}
} else if (tmp.charAt(0) == 'ε') {
stack.pop();
show("" + ctmp + "->" + 'ε');
} else {
stack.pop();
stack.push(tmp);
show("" + ctmp + "->" + tmp);
}
}
}
//过滤方法将打印的字符串中e和t替换为E'和T'
public String filter(String src) {
if(src.contains("e") || src.contains("t")) {
StringBuffer result = new StringBuffer();
char item;
for(int i = 0;i < src.length(); i++) {
item = src.charAt(i);
if(item == 'e') {
result.append("E'");
continue;
}else if(item == 't') {
result.append("T'");
continue;
}
result.append(item);
}
return result.toString();
}else {
return src;
}
}
public String work(String inputExpression) {
input = inputExpression + '#';
symbol = true;
tempBuffer.append(String.format("%-8s%-20s%-20s%-20s\r\n",
"步骤","分析栈","剩余输入栈","所用产生式"));
analyse();
if (symbol) {
tempBuffer.append("\r是正确的符号串\r");
return tempBuffer.toString();
} else {
tempBuffer.append("\r不是正确的符号串\r");
return tempBuffer.toString();
}
}
public StringBuffer getTempBuffer() {
return tempBuffer;
}
public void setTempBuffer(StringBuffer tempBuffer) {
this.tempBuffer = tempBuffer;
}
public Stack getStack() {
return stack;
}
public void setStack(Stack stack) {
this.stack = stack;
}
public String[][] getTab() {
return tab;
}
public void setTab(String[][] tab) {
this.tab = tab;
}
public String getInput() {
return input;
}
public void setInput(String ns) {
this.input = ns;
}
public int getPtr() {
return ptr;
}
public void setPtr(int ptr) {
this.ptr = ptr;
}
public int getRow() {
return row;
}
public void setRow(int row) {
this.row = row;
}
public int getCol() {
return col;
}
public void setCol(int col) {
this.col = col;
}
public int getStep() {
return step;
}
public void setStep(int step) {
this.step = step;
}
public boolean isBoo() {
return symbol;
}
public void setBoo(boolean boo) {
this.symbol = boo;
}
}
三、主程序GrammarMain.java
package grammarAnalyze;
import java.util.Scanner;
public class GrammarMain {
public static void main(String[] args){
boolean isContinue = true;
while(isContinue) {
GrapparAnalyze analyze = new GrapparAnalyze();
Scanner scan = new Scanner(System.in);
System.out.println("请输入你要翻译的表达式:");
String inputExpression = scan.nextLine();
String srcdata = inputExpression.trim();
if("".equals(srcdata) || srcdata == null) {
System.out.println("输入表达式为空,请重新输入");
}else {
String result = analyze.work(srcdata);
System.out.println(result);
System.out.println("是否继续?y/n");
String yn = scan.nextLine();
if("no".equals(yn) || "n".equals(yn)) {
isContinue = false;
}
}
}
}
}
四、测试运行


java编写简单的语法分析预测程序的更多相关文章
- 编写简单的spring mvc程序,在tomcat上部署
编写简单的spring mvc程序,在tomcat上部署 1 用java 配置spring mvc ,可以省去web.xmlpackage hello;import org.springframewo ...
- VC++编写简单串口上位机程序
VC++编写简单串口上位机程序 转载: http://blog.sina.com.cn/s/articlelist_1809084904_0_1.html VC++编写简单串口上位机程序 串口通信 ...
- 用Python编写简单的发红包程序和计算器原理
用Python编写简单的发红包程序: 第一种解法:数轴方法解决 import random def red_packet(money,num): money = money * 100 #将钱数转换成 ...
- java编写简单的累加程序
编程思路:1.建立类包demo: 2.在类包中建立CommanParameter类: 3.利用for循环通过强制类型转换将在后台中输入的String类型的字符转换为整型并进进累加操作: package ...
- 使用Java编写并运行Spark应用程序
我们首先提出这样一个简单的需求: 现在要分析某网站的访问日志信息,统计来自不同IP的用户访问的次数,从而通过Geo信息来获得来访用户所在国家地区分布状况.这里我拿我网站的日志记录行示例,如下所示: 1 ...
- Java编写高质量代码改善程序的151个建议
第一章 Java开发中通用的方法和准则 建议1:不要在常量和变量中出现易混淆的字母: (i.l.1:o.0等). 建议2:莫让常量蜕变成变量: (代码运行工程中不要改变常量值). 建议3:三元操作符 ...
- 利用Java编写简单的WebService实例
使用Axis编写WebService比較简单,就我的理解,WebService的实现代码和编写Java代码事实上没有什么差别,主要是将哪些Java类公布为WebService. 以下是一个从编写測试样 ...
- 利用Java编写简单的WebService实例-转载
使用Axis编写WebService比较简单,就我的理解,WebService的实现代码和编写Java代码其实没有什么区别,主要是将哪些Java类发布为WebService.下面是一个从编写测试例子到 ...
- 用Java实现简单的网络聊天程序
Socket套接字定义: 套接字(socket)是一个抽象层,应用程序可以通过它发送或接收数据,可对其进行像对文件一样的打开.读写和关闭等操作.套接字允许应用程序将I/O插入到网络中,并与网络中的其他 ...
随机推荐
- D. Magic Box(几何)
One day Vasya was going home when he saw a box lying on the road. The box can be represented as a re ...
- docker kafka 修改hostname导致的问题
昨天发现开发环境的3台kafka无法消费,所以今日kafka的容器执行如下语句 bash-4.4# ./kafka-topics.sh --describe --zookeeper 192.168.0 ...
- 关于jxl的getCellFormat()方法获取表格样式----中文货币乱码
File templateFile = getTempalte(client.getSc_shortName());//这里读取模板文件 WorkbookSettings set1 = new Wor ...
- [译文]casperjs 的API-casper模块
Casper class: 可以通过这个模块的create()方法来获取这个模块的一个实例,这是最容易的: var casper = require('casper').create(); 我们也可以 ...
- 题目1022:游船出租(hash简单应用)
问题来源 http://ac.jobdu.com/problem.php?pid=1022 问题描述 每次输入:船号(1~100) 键值(S或E) 发生时间(小时:分钟).当船号为0时,代表一天结束: ...
- QRCode简介(收藏)
一.什么是二维码:二维码 (2-dimensional bar code),是用某种特定的几何图形按一定规律在平面(二维方向上)分布的黑白相间的图形记录数据符号信息的. 在许多种类的二维条码中,常 ...
- 微信小程序通过background-image设置背景图片
微信小程序通过background-image设置背景:只支持线上图片和base64图片,不支持本地图片:base64图片设置步骤如下:1.在网站http://imgbase64.duoshitong ...
- 基础篇:1)时代的发展与结构设计--3d与2d设计的变迁
本章目的:机械设计需要追寻时代的脚步. 1.机械作图时代的划分 这张图是作者个人对机械作图发展的简单划分,并非哪个网站的数据.所以粗糙或错误请海涵. 2.2d作图时代 纯2d时代中,工程师开只绘制二维 ...
- element-ui table多选CheckBox参数解析
element-UI里的table表格与多选框CheckBox的组合很常用,官网也给了很多参数,自己总结了一下,方便日后使用 1.简易用法,没有附加的功能 要在表格里使用CheckBox很简单,只需设 ...
- thinkPhP + Apache + PHPstorm整合框架
最近在学习使用 ThinkPhP,网上很多都是用一些整合好的服务框架,为了学习,在这里我简单的对Apache.PHP做一个原生的整合,希望对你有帮助. 步骤: ①下载 thinkPHP.PHP.Apa ...