目录
第1关:递归分析法设计与实现
任务描述
本关任务:根据某一文法编制调试递归下降分析程序,以便对任意输入的符号串进行分析。
相关知识
为了完成本关任务,你需要掌握:递归下降分析程序设计与实现。
递归分析法
递归下降分析法,顾名思义就是使用递归的思想去分析。对于一个文法
G
,对其每一个非终结符U构造一个递归过程,一般的,以非终结符的名字来命名这个子过程。所有子程序构造完成后,对指定文法,运行文法开始符号对应的子程序,返回匹配结果。递归下降分析法是一种简洁的分析方法,对于给定的文法
G
。S -> N V N N -> s | t | g | w V -> e | d
可以根据
S
的产生式,得到N,V,N
三个非终结符的子程序,在子程序中,可以分别对N,V,N
进行相关的定义:parse_S() parse_N() parse_V() parse_N() parse_N() token = tokens[i++] if(token == s || token == t || token == g || token == w) return; error("...") parse_V() token = tokens[i++] ... //
同样的,如果
N,V
的产生式中还具有非终结符,我们便可以继续递归,完成文法的G
的分析。实验步骤
根据上述可知,在使用递归分析法完成语法分析时,需要做到:
对语法规则有明确的定义;
在递归分析法中,文法由程序之间的相互调用形成递归而构成,我们需要在每个子程序中给出该符号的相关选择就可以了,例如,对于某一符号X,可以给出如下的定义:void X() { if (str[index] == '+') { index++; T(); X(); } }
编写的分析程序能够进行正确的语法分析;
对于遇到的语法错误,能够做出简单的错误处理,给出简单的错误提示,保证顺利完成语法分析过程;
编程要求
根据提示,在右侧编辑器Beign和End之间补充判断为
+
或者-
号、判断*
号和/
号、判断字符
和()
的代码,点击测评,运行程序,进行结果对比。根据下列文法完成程序设计,并给出相应提示:
(1)E- TG (2)G- +TG|—TG (3)G- ε (4)T- FS (5)S- *FS|/FS (6)S- ε (7)F- (E) (8)F- i
输出的格式如下:
程序会自动读取预设的输入完成类似的输出:
请在键盘键入表达式:(以#号结束) i+i/i# +i/i# i/i# /i# i# # 表达式正确!
答案:
import java.io.*; import java.util.Scanner; public class ex3 { public static String buffer = null;//存取从键盘输入的字符 public static int p = 0;//字符当前的位置 public static String ch;//存取当前字符 public static boolean error = false;//判断表达式是否正确 public static int m = 0; /*******Beign*******/ /*判断是否为+或者-号*/ public boolean plusandminus()// { if (multiplyanddivide()){ while (ch.equals("+")||ch.equals("-")){//判断+号和-号 ch = buffer.substring(++p, p+1);//获取当前字符 outputspace(p + 1, m); if (!multiplyanddivide()){ error=true; return false; } } }else{ error=true; return false; } return true; } /********End********/ /*******Beign*******/ /*判断*号和/号*/ public boolean multiplyanddivide()// { if(brackets()){ while(ch.equals("*")||ch.equals("/")){//判断*号和/号 ch=buffer.substring(++p,p+1);//获取当前字符 outputspace(p+1,m); if(!brackets()){ error=true; return false; } } }else{ error=true; return false; } return true; } /********End********/ /*******Beign*******/ /*字符和”()“的判断*/ public boolean brackets()// { if(ch.equals("(")){//判断左括号 ch=buffer.substring(++p,p+1);//获取当前字符 outputspace(p+1,m); if(!plusandminus()){ return false; } if(ch.equals(")")){//判断右括号 ch=buffer.substring(++p,p+1);//获取当前字符 outputspace(p+1,m); return true; }else{ error=true; return false; } }else if(ch.matches("[a-zA-Z]")){//判断26个英文字母大小写 ch=buffer.substring(++p,p+1);//获取当前字符 outputspace(p+1,m); return true; }else{ error=true; return false; } } /********End********/ public void outputspace(int n,int m)//输出空格 { StringBuffer buf = new StringBuffer(); for(int i = 0;i < n;i++) { buf.append(" "); } if(n<m) { System.out.print(buf.toString()); System.out.println(buffer.substring(p+1,m)); } } public void mainoperation(int n,int a) throws IOException//读取输入字符,如果出错,抛出IO异常 { p = 0; boolean flag = true; System.out.println("请在键盘键入表达式:(以#号结束)"); BufferedReader buf = new BufferedReader(new InputStreamReader(System.in));//定义输入流 buffer = buf.readLine();//读取一行字符 System.out.println(buffer); m=buffer.length(); while(flag) { ch = buffer.substring(p,p+1); outputspace(p+1,m); if(ch.equals("#") || error)//判断是否结束 { flag = false; break; } plusandminus();//子程序判断表达式是否正确 } if(error) { System.out.println("表达式错误!"); } else { System.out.println("表达式正确!"); } if(n == a + 1) buf.close();//关闭流对象 } public static void main(String[] args) throws IOException { int n; ex3 Re = new ex3(); n = 1; for(int a = 0;a < n;a++) { Re.mainoperation(n,a);//主控程序 } } }