合肥工业大学编译原理实验_第1页
合肥工业大学编译原理实验_第2页
合肥工业大学编译原理实验_第3页
合肥工业大学编译原理实验_第4页
合肥工业大学编译原理实验_第5页
已阅读5页,还剩23页未读 继续免费阅读

下载本文档

版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领

文档简介

1、宣城校区实 验 报 告课 程 名 称 编译原理 专 业 班 级 计算机0001班 学生姓名及学号 赵保飞 2015216768 指 导 教 师 李芒宏 实 验 地 点 计算机中心楼第四机房 2017 2018 学年第 一 学期编译原理课程实验报告实验名称词法分析设计姓 名赵保飞系院专业计算机科学与技术班级计算机01班学号2015216768实验日期2017.10.18指导教师李芒宏成绩一、实验目的和要求通过本实验的编程实践,使学生了解词法分析的任务,掌握词法分析程序设计的原理和构造方法,使学生对编译的基本概念、原理和方法有完整的和清楚的理解,并能正确地、熟练地运用。二、实验原理(1)实验数据结

2、构说明KString数组-关键字表;schar数组分界符;mchar数组算术运算符;rString数组关系运算符;ArrayList型String数组ci常数;ArrayList型String数组id标识符(2)实验算法描述(3)算法流程图三、源程序代码和测试结果package lexicalAnalysis;import java.util.*;import java.io.*;public class lexicalAnalysis static String k="for","main","if","while&quo

3、t;,"void","public","static","printf","scanf","asm","do","return","typedef","auto","double","break","short","using","default","long"/关键字stati

4、c char s=',','','(',')','','','',''/2分界符static char m='+','-','*','/'/3算术运算符static String r="<","<=","=",">",">=","<>"/4关系运算符A

5、rrayList<String> ci=new ArrayList<String>();/5常数ArrayList<String> id=new ArrayList<String>();/6标识符String tempToken=""/临时存放组成一个“词”单位串int pint ,row = 1,line = 1;/当前指针指示,行数,列数char ch;/存放最新读入源程序字符String instring;/存放输入de源程序代码public static void main(String args)throws Exc

6、eption/ TODO Auto-generated method stublexicalAnalysis one = new lexicalAnalysis();System.out.println("单词"+"t二元序列"+"t类 型"+"t位置(行,列)");one.readtext();boolean isdigit(char c)/判断所读字符是否为数字,是则返回ture,否则返回falseif(c>=48 && c<=57 )return true;elsereturn f

7、alse;boolean isletter(char c)/判断所读字符是否为字母,是则返回true,否则返回falseif(c>64&&c<91)|(c>96&&c<123)return true;elsereturn false;boolean isline(char c)/判断字符c是否是下划线"_" if(c='_') return true; else return false; boolean remove()/用于在判断关系运算符时,判断是否是要再读一个字符 char b=instring

8、.charAt(pint+1);/string类charAt() 方法用于返回指定索引处的字符。索引范围为从 0 到 length()-1if(b='='|b='>')/当其后的字符是=或>时,要再读一个字符。否则不要再读。 return true;else return false; void clearBlank()/检查空白直到读入字符非空白 while(ch=' ') getchar(); void getchar()pint+;if(pint <= instring.length()-1)ch=instring.cha

9、rAt(pint);if(ch = 'n')change_row_line();getchar();elsech=' '/instring数据读取完,需重新读入void retract()/读入前一个字符 pint-; ch=instring.charAt(pint); void distinguishLeter()/识别字符串tempToken=""/清空while(isletter(ch) | isdigit(ch) | isline(ch)/字母,数字,下划线仍是字符串合法组成,继续识别tempToken=tempToken + ch;

10、getchar();void distinguishDigit()/识别数字串 tempToken=""/先将strtoken置空 while(isdigit(ch)/当数字时继续识别数字串 tempToken=tempToken+ch;/将新识别的字符加到strtoken后 getchar(); if(isletter(ch)|isline(ch)/识别完数字串而其后是字母,下划线时出错处理 while(isletter(ch)|isline(ch)|isdigit(ch)/当是字母,数字,下划线时继续识别错误数字串 tempToken=tempToken+ch;/将新识

11、别的字符加到strtoken后 getchar(); display(0,tempToken,' ');/输出错误数字串 tempToken=""/将strtoken置空返回 void display(int i,String s,char a)/各种输出处理 switch(i) case -1:System.out.println(a+"tError"+"tError"+"t"+row+","+line+")"); break; case 0:System.

12、out.println(s+"tError"+"tError"+"t("+row+","+line+")"); break; case 1:System.out.println(s+"t(1,"+s+")"+"t关键字 "+"t("+row+","+line+")"); break; case 2:System.out.println(a+"t(2,"+a+

13、")"+"t分界符"+"t("+row+","+line+")"); break; case 3:System.out.println(a+"t(3,"+a+")"+"t算术运算符"+"t("+row+","+line+")"); break; case 4:System.out.println(s+"t(4,"+s+")"+"

14、t关系运算符 "+"t("+row+","+line+")"); break; case 5:System.out.println(s+"t(5,"+s+")"+"t常数"+"t"+"("+row+","+line+")"); break; case 6:System.out.println(s+"t(6,"+s+")"+"t标识符&qu

15、ot;+"t("+row+","+line+")"); break; line+;/列数加一 void change_row_line()/改变行数和列数row+;line=1;void handleString()/输入串处理pint=-1;/将搜索指示器置-1System.out.println("要处理的语句为 : "+instring);getchar();/读入一个字符while(pint<instring.length()/当搜索指示器没有越界时clearBlank();/检查空白直到读入读入非空

16、if(isdigit(ch)/当ch为数字时进行数字串识别distinguishDigit();/数字串识别if(tempToken.length()!=0)/经过数字串识别后,如果strtoken不为空if(reseverci(tempToken)=-1)/如果strtoken不在ci表中,将strtoken加入ci表中ci.add(tempToken);/将strtoken加入ci表中display(5,tempToken,' ');/输出数字串else /如果strtoken在ci表中,仅输出display(5,tempToken,' ');/输出数字串e

17、lse if(isletter(ch)/当ch为字母时进行字符串识别distinguishLeter();/字符串识别if(reserve(tempToken)=-1)/如果strtoken不在k表中if(reseverid(tempToken)=-1)/如果strtoken不在id表中id.add(tempToken);/将strtoken加入id表中display(6,tempToken,' ');/输出标识串else /如果strtoken在id表中display(6,tempToken,' ');/输出标识串else /如果strtoken在关键字表di

18、splay(1,tempToken,' ');/输出关键字else if(in_s(ch)!=-1)/分界符处理包含在one.in_s(one.ch)中 else if(in_m(ch)!=-1)/算术运算符处理包含在one.in_m(one.ch)中else if(in_k(ch)!=-1)/关系运算符处理包含在one.in_k(one.ch)中elsedisplay(-1, "",ch);/errorgetchar();/读下一位int reserve(String s)/判断字符串是否是保留字int i;for(i=0;i<k.length;i+

19、)if(s.equals(ki)return i;/是保留字,就返回编码return -1;/不是保留字,就返回-1int reseverid(String s)/判断识别的标志符是否已经在id表中int i;for(i=0;i<id.size();i+)if(s.equals(id.get(i)return i;/识别的标志符已经在id表中,返回位置return -1;/识别的标志符不在id表中,返回-1int reseverci(String s)/判断识别的数字串是否已经在ci表中int i;for(i=0;i<ci.size();i+)if(s.equals(ci.get(

20、i)return i;/识别的数字串已经在ci表中,返回位置return -1;/识别的数字串不在ci表中,返回-1int in_s(char c)/确认分界符int i;for(i=0;i<s.length;i+)if(c=si)/与某个分界符配备时display(2, "",c);/输出分界符getchar();/读下一位return i;/返回所在位置return -1;/不在分界符表中int in_m(char c)/查找算术运算符int i;for(i=0;i<m.length;i+)if(c=mi)/与某个算术运算符配备时display(3, &qu

21、ot;",c);/输出算术运算符getchar();/读下一位return i;/返回所在位置return -1;/不在算术运算符表中int in_k(char b)/查找关系运算符int i;if(remove()=false)/读下一位为假时,进行一位关系运算符识别for(i=0;i<r.length;i+)if(ri.length()=1)/当关系运算符为一位时,尝试匹配 if(ri.equals(Character.toString(b)display(4, ri,' ');/输出关系运算符getchar();/读下一位return i;/返回所在位置e

22、lse/读下一位为假时,进行两位关系运算符识别char a=new char2;/将两位字符放入a中a0=b;getchar();a1=ch;for(i=0;i<r.length;i+)if(ri.length()=2)/当关系运算符为两位时,尝试匹配if(ri.equals(String.copyValueOf(a)display(4, ri,' ');/输出关系运算符getchar();/读下一位return i;/返回所在位置retract();/两位关系运算符匹配失败,读入前一个字符return -1;/不在关系运算表中void readtext()throws

23、Exception/从文本读入源程序String pathname = "C:UsersYodaDesktopcode.txt"File filename = new File(pathname);InputStreamReader reader = new InputStreamReader(new FileInputStream(filename);BufferedReader br = new BufferedReader(reader);instring = ""instring=br.readLine();while(instring!=nul

24、l)handleString();/处理读入的源程序change_row_line();instring=br.readLine();四、实验评价、收获与体会纸上得来终觉浅,虽然词法分析器是理论上实行最简单的,但是编写程序实现时却也遇到了不少的问题。使用JAVA来写是因为这学期正在开这门课,数据的读入还是自己提前看才实现的,读入的一条源代码看似简单,首先要储存,储存后要清楚空格直到读到字符为止,还是以C+的思维来写java程序,其中很多细节均是通过函数来实现。这也许自己没有深刻理解java设计思维的原因。希望以后能改进。编译原理课程实验报告实验名称LL(1)分析法姓 名赵保飞系院专业计算机科学

25、与技术班级计算机01班学号2015216768实验日期10.25指导教师李芒宏成绩一、实验目的和要求复变函数。编译原理实验和预习。java作业和实验。操作系统复习。通过完成预测分析法的语法分析程序,了解预测分析法和递归子程序法的区别和联系。使学生了解语法分析的功能,掌握语法分析程序设计的原理和构造方法,训练学生掌握开发应用程序的基本方法。有利于提高学生的专业素质,为培养适应社会多方面需要的能力。二、实验原理文法为(1)E->TG(2)G->+TG|-TG(3)G->(4)T->FS(5)S->*FS|/FS(6)S->(7)F->(E)(8)F->

26、;i(1)实验数据结构说明Char数组-Vn数组-非终结符表;char数组-Vt数组-终结符;String数组-Gr数组-文法;Boolean型FIRST二维数组对应每个非终结符的first集,初始化均为false;Boolean型FOLLOW二维数组对应每个非终结符的Follow集,初始化均为false;String型M二维数组对应预测分析表(2)实验算法描述算法总的来讲不是很复杂,但是细节上可能有点复杂。总的是依据本实验的数据结构FIRST和FOLLOW集,这两个数据结构虽然浪费了不少空间但是确实让程序的实现变得更简单。First()总的来求所有的非终结符的first集,first1()分

27、工为具体求某一个具体非终结符的first集;first2()则是求某一个集体的产生式的first集。通过这三个函数的逐级分工来实现求所有的非终结符的first集。Follow集来说相对难求一点,但把它放到first集后面求来说相对简单一点。通过上面的分布设计就可以实现了。(3)算法流程图三、源程序代码和测试结果package exp3;import javax.swing.*;import java.awt.*;import java.awt.GridLayout.*;import java.awt.event.ActionEvent;import java.awt.event.ActionL

28、istener;class WinGrid extends JFrameGridLayout grid ;JPanel chessboard;JTextField text; JTextArea textShow; JButton button; ReaderListen listener;WinGrid()init();setVisible(true); setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE); void init() setLayout(new FlowLayout(); text = new JTextField(10); setBo

29、unds(466,166,500,400); button = new JButton("读取"); textShow = new JTextArea(9,30); listener = new ReaderListen(); listener.setJTextField(text); listener.setJTextArea(textShow); text.addActionListener(listener); button.addActionListener(listener); add(text); add(button); add(new JScrollPane

30、(textShow); class ReaderListen implements ActionListener JTextField text; JTextArea textShow; LL one=new LL(); String s;String temp1= new String186; public void setJTextField(JTextField text) this.text = text; public void setJTextArea(JTextArea textShow) this.textShow = textShow; public String S() S

31、tring t = text.getText(); return t; Override public void actionPerformed(ActionEvent e) try String s = text.getText()+"#" textShow.append(s+"n"); textShow.append("步骤"+"t"+"分析栈"+"t"+"剩余输入串"+"t"+"所用产生式"+"t&q

32、uot;+"动作"+"n"); one.right(s,temp1); for(int i = 0;i<18;i+) for(int k=0;k<5;k+) textShow.append(temp1ik+"t"); textShow.append("n"); catch (Exception e2) e2.printStackTrace(); public class exp3 public static void main(String args) / TODO Auto-generated met

33、hod stubnew WinGrid();package exp3;import java.util.*;import java.io.*;public class LL char Vn='E','T','G','F','S'/非终结符char Vt='+','-','*','/','(',')','i','#',''/终结符boolean FIRST=new boole

34、anVn.lengthVt.length+1;/bool型数组,初始化均为flase boolean FOLLOW=new booleanVn.lengthVt.length+1;/bool型 String M=new StringVn.lengthVt.length-1;/预测分析表String Gr="E->TG","G->+TG|-TG","G->","T->FS","S->*FS|/FS","S->","F->(E)

35、","F->i"/文法LL()first();/求first集 follow();/求follow集 buildM();/求预测分析表 int or(int i,String s)/返回该文法第i符号离他最近的在其右边一个“|”位置 for(i=i+1;i<s.length();i+) if(s.charAt(i)='|') return i;/存在,就返回位置 return -1;/返回-1表示没有“|”在其右边 int vnNum(char c)/返回c在非终结符表中的位置 int i; for(i=0;i<Vn.length

36、;i+) if(c=Vni) return i;/在表中,就返回位置 return -1;/返回-1表示不在表中 int vtNum(char c)/返回c在终结符表中的位置 int i; for(i=0;i<Vt.length;i+) if(c=Vti) return i; return -1; void first2(String s, int j)/求关于某一个产生式s,关于第j个字符的的first集 int v=vnNum(s.charAt(0);/v是产生式左边的非终结符所在序号 int i; if(vtNum(s.charAt(j)!=-1)/若产生式右边第一个为终结符 FI

37、RSTvvtNum(s.charAt(j)=true;/就把s.charAt(j)加入s.charAt(0)的first集 else/产生式右边第一个为非终结符if(!FIRSTvnNum(s.charAt(j)Vt.length)first1(s.charAt(j);for(i=0;i<Vt.length;i+)if(FIRSTvnNum(s.charAt(j)i && Vti!='')FIRSTvi=true; if(vtNum('')!=-1)/终结符中有if(FIRSTvnNum(s.charAt(j)vtNum(''

38、;)if(j = s.length()-1)FIRSTvvtNum('')=true;return ;if(s.charAt(j+1)!='|')j+;first2(s,j);elseFIRSTvvtNum('')=true; void first1(char v)/求非终结符v关于该文法的first集 int i,j; String s; for(i=0;i<Gr.length;i+) s=Gri; if(s.charAt(0) = v) j=3; first2(s,j); while(or(j,s)!=-1&&j<

39、s.length() j=or(j,s); first2(s,j+1); FIRSTvnNum(v)Vt.length=true;/将fivn(v)Vt.length设为true,表示已求v的first集 void first()/求所有非终结符的first集 int i,j; for(i=0;i<Vn.length;i+)/遍历非终结符表求各自first集 if(!FIRSTiVt.length) first1(Vni); void follow2(String s,int j)/求关于某一个产生式的follow集 int i; if(j=s.length()-1|(j<s.le

40、ngth()-1&&s.charAt(j+1)='|') if(s.charAt(0)!=s.charAt(j)/考察字符与产生式左边的非终结符不同时 if(!FOLLOWvnNum(s.charAt(0)Vt.length) follow1(s.charAt(0); for(i=0;i<Vt.length;i+)/产生式左边的非终结符的follow集加到考察字符的follow集中 if(FOLLOWvnNum(s.charAt(0)i) FOLLOWvnNum(s.charAt(j)i=true; if(j<s.length()-1&&am

41、p;s.charAt(j+1)!='|') if(vtNum(s.charAt(j+1)!=-1)/该字符为终结符 if(VtvtNum(s.charAt(j+1)!='') FOLLOWvnNum(s.charAt(j)vtNum(s.charAt(j+1)=true; else for(i=0;i<Vt.length;i+)/该字符的first集中除的非终结符加入考察字符的follow集中 if(FIRSTvnNum(s.charAt(j+1)i&&Vti!='') FOLLOWvnNum(s.charAt(j)i=tr

42、ue; if(vtNum('')!=-1)/非终结符中有 if(s.charAt(0)=s.charAt(j) return; boolean m=true;/当考察字符右边的字符串的first集中有'',m为真,没有时,m为假 for(i=j+1;i<s.length();i+) if(vtNum(s.charAt(i)!=-1) m=false; break; if(s.charAt(i)='|') break; if(!FIRSTvnNum(s.charAt(i)vtNum('')/当考察字符右边的字符串中的有一非终结

43、符的first集中不含,m为假 m=false; if(m) if(!FOLLOWvnNum(s.charAt(0)Vt.length) follow1(s.charAt(0); for(i=0;i<Vt.length;i+)/产生式左边的非终结符的follow集加到考察字符的follow集中 if(FOLLOWvnNum(s.charAt(0)i) FOLLOWvnNum(s.charAt(j)i=true; void follow1(char v)/求非终结符v关于该文法的follow集 if(v='E')/v为开始符号时 FOLLOWvnNum(v)vtNum(&#

44、39;#')=true; int i,j; String s; for(i=0;i<Gr.length;i+) s=Gri; for(j=3;j<s.length();j+) if(s.charAt(j)=v)/产生式右边有考察字符 follow2(s,j);/求关于该产生式的follow集 FOLLOWvnNum(v)Vt.length=true; void follow()/求所有非终结符的follow集 int i,j; for(i=0;i<Vn.length;i+)/非终结符的follow集未求时 if(!FOLLOWiVt.length) follow1(V

45、ni); void MM(int j,int i,String s,int m)/将某一个产生式填入预测分析表中 char u=Gri.charAt(0); int k; if(vtNum(Gri.charAt(j)!=-1)/为终结符if(Gri.charAt(j)!='')/Gri.charAt(j)不为时将s加到Mvn(u)vt(Gri.charAt(j)中MvnNum(u)vtNum(Gri.charAt(j)=s;else/是 for(k=0;k<Vt.length-1;k+)/Gri.charAt(j)为,将属于u的follow集的元素b的位置用s加到Mvn(

46、u)vt(b)中if(FOLLOWvnNum(u)k)MvnNum(u)k=s; else/Gri.charAt(j)为非终结符for(k=0;k<Vt.length-1;k+)/对于终结符a属于Gri.charAt(j)的first集时,将s加到Mvn(u)vt(a)中if(FIRSTvnNum(Gri.charAt(j)k)MvnNum(u)k=s;if(FIRSTvnNum(Gri.charAt(j)vtNum('')/当属于Gri.charAt(j)的first集时,将属于u的follow集的b将s加到Mvn(u)vt(b)中if(j=m-1)for(k=0;k&

47、lt;Vt.length-1;k+)if(FOLLOWvnNum(u)k)MvnNum(u)k=s;elsej=j+1; MM(j,i,s,m); void buildM()/构造预测分析表 int i,j,m; String s; for(i=0;i<Gr.length;i+) j=3; while(j<Gri.length() m=or(j,Gri); if(m=-1) m=Gri.length(); s=Gri.substring(j, m);/将j到m-1赋值到s 截取一段字符(从索引出发) MM(j,i,s,m); j=m+1; void right(String s,S

48、tring t)/分析一个字符串是否符合该文法 Stack<Character> temp=new Stack<Character>();/分析栈 String tempp = t;/保存输出临时数组 String z; temp.setSize(20); temp.push(new Character('#');/初始化将#入栈 temp.push(new Character('E');/初始化将E入栈 char u,v; int i=0,j,k=0; String m,action="初始化",rule="

49、; " while(i < s.length() temppk0 = String.valueOf(k);/整数型变量赋值给字符串数组使用函数String.valueOF() u=s.charAt(i); String temp0_string ="" for(j=0;j<temp.size();j+) if(temp.get(j)!=null) temp0_string = temp0_string.concat(temp.get(j).toString(); temppk1 = temp0_string; temppk2 = s.substring(i); temppk3 = rule; temppk4 = action.toString(); v=temp.pop(); action="pop" if(vnNum(v)!=-1)/栈顶元素为非终结符时 if(MvnNum(v)vtNum(u)!=null)/分析表中有产生式 m=MvnNum(v)vtNum(u); rule=v+"->"+m; if(!m.equals("&qu

温馨提示

  • 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
  • 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
  • 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
  • 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
  • 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
  • 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
  • 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

评论

0/150

提交评论