《编译原理》课程设计报告-LALR(1)分析器.doc_第1页
《编译原理》课程设计报告-LALR(1)分析器.doc_第2页
《编译原理》课程设计报告-LALR(1)分析器.doc_第3页
《编译原理》课程设计报告-LALR(1)分析器.doc_第4页
《编译原理》课程设计报告-LALR(1)分析器.doc_第5页
免费预览已结束,剩余27页可下载查看

下载本文档

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

文档简介

编译原理课程设计报告 题目:lalr(1)分析器 2011年6月一 设计目的 1 巩固对语法分析的基本功能和原理的认识。2 通过对语法分析表的自动生成加深语法分析表的认识。3 理解并处理语法分析中的异常和错误。4 熟练掌握lalr(1)语法分析过程。二 设计内容本次课程设计是设计一个lalr(1)语法分析器。lalr(1)是lr(1)的一种改进。为了克服lr(1)中分析表的构造对某些同心集的分裂可能对状态数目引起剧烈的增长,从而导致存储容量的急剧增加,采用对lr(1)项目集规范族合并同心集的方法,若合并同心集后不产生新的冲突,则为lalr(1)项目集。本次课程设计的主要内容有首先生成lr(1)项目族;再合并同心集;然后生成lalr(1)项目族,求出lalr(1)的分析表;最后能判断一个字符串是否是该文法的串,若是则生成该字符串的树。具体实现过程及相关主要实现类如下:类:lr0功能:lr0核心项目集构造输入:产生式的数字化表示输出:lr0核心项目集实现思想:lr0核心项目集构造算法程序如下:public class lr0arraylistarraylist core;arraylistarraylist ary_c;arraylist producer;private int copy(int a )int val=new inta.length;for(int i=0;ia.length;i+)vali=ai;return val;private arraylist copy(arraylista)arraylist val=new arraylist ();for(int i=0;ia.size();i+)val.add(copy(a.get(i);return val;public boolean isequal(int a,int b)if(a.length!=b.length)return false;for(int i=0;ia.length;i+)if(ai!=bi)return false;return true;public boolean isequal(arraylist a,arraylistb)if(a.size()!=b.size()return false;for(int i=0;ia.size();i+)if(!isequal(a.get(i),b.get(i)return false;return true;public int indexof(arraylist a,int b)int index;int s1;for(index=0;indexa.size();index+)s1=a.get(index);if(isequal(s1,b)=true)return index;return -1;public int indexof(arraylist arraylist a,arraylist b)int index;arraylist s1;for(index=0;indexa.size();index+)s1=a.get(index);if(isequal(s1,b)=true)return index;return -1;public int getstartwith(int i)int count=i;arraylist temp=new arraylist(3);while(counti)break;count+;int val=new inttemp.size();for(count=0;counttemp.size();count+)valcount=temp.get(count);return val;public void closuer_innal(int item,arraylist pos)if(item.length=2)return ;int i;int p;for(i=0;iitem.length&itemi!=-1;i+);if(item.length-1=i)return ;if(itemi+11000)p=getstartwith(itemi+1);for(i=0;ip.length;i+)if(pos.indexof(pi)=-1)pos.add(pi);closuer_innal(producer.get(pi),pos);return ;public arraylist closure(int a) /求闭包arraylist val=new arraylist();arraylist pos=new arraylist();closuer_innal(a,pos);for(int i=0;ipos.size();i+)val.add(copy(producer.get(pos.get(i);return val;public arraylist goto(arraylista,int x)int index1=0,index2=0,index3,bl,index=0;int s1,s2; arraylist temp1 = new arraylist (); /临时arraylist temp2 = new arraylist ();temp1=copy(a);s1=temp1.get(index1);if(s1=null)return null;while(s1index2!=-1)index2+;if(s1.length=index2)return null;index2+;for(index3=0;index3temp1.size();index3+)s1=copy(temp1.get(index3);if(s1.lengthindex2+1) /防止array溢出continue;if(s1index2=x&indexof(temp2,s1)0)bl=s1index2-1;s1index2-1=s1index2;s1index2=bl; /-1往后移位temp2.add(copy(s1);bl=temp2.size();for(index=0;indexbl;index+) /求goto函数中的返回值的closure集s1=copy(temp2.get(index);temp1=closure(s1);for(index1=0;index1temp1.size();index1+)s2=temp1.get(index1);if(indexof(temp2,s2)0)temp2.add(s2);return temp2;public lr0(arraylist p)core =new arraylistarraylist();ary_c=new arraylistarraylist();producer=p;public void total()arraylist cur=new arraylist();cur.add(producer.get(0);core.add(cur);int i=0;while(icore.size()cur=core.get(i);lr0split(cur);i+;update();public void lr0split(arraylistcur)arraylist handle=copy(cur);arraylist tempal;int temp;int prod;int start;arraylist result=new arraylist();arraylist val=new arraylist();int j;for(int i=0;icur.size();i+)tempal=closure(cur.get(i);for(j=0;jtempal.size();j+)if(indexof(handle,tempal.get(j)=-1)handle.add(tempal.get(j);while(!handle.isempty()prod=(int) handle.get(0);start=getstart(prod); /返回-1之后的数,没有返回-1if(start=-1)handle.remove(0);continue;result=getstartwith(start,handle); /返回handle中所有-1之后是start的产生式,并删除repair(result); /将result中产生式-1以后的数与-1调换 /*returnval=new arraylist();for(int i=0;iresult.size();i+)temp=(int) result.get(i);val=closure(temp);if(indexof(returnval,temp)0);returnval.add(temp);for(int j=0;jval.size();j+)if(indexof(returnval,val.get(j)0);returnval.add(val.get(j);*/if(indexof(core,result)0)core.add(result);public int getstart(int prod) /返回-1之后的数,没有返回-1int i=0;while(prodi!=-1&iprod.length)i+;if(prod.length-1=i)return -1;else return prodi+1;public arraylist getstartwith(int start, arraylisthandle) arraylist val=new arraylist() ;int i;for(i=0;ihandle.size();i+)if(isstartwith(start,handle.get(i) val.add(handle.get(i);handle.remove(i);i-;return val; /返回handle中所有-1之后是start的产生式,并删除public boolean isstartwith(int start,int prod) /产生式-1之后是否为start,是返回1,否则返回0;boolean state=false;if(getstart(prod)=start)state=true;return state;public void repair(arraylist prod) /将result中产生式-1以后的数与-1调换 int num;int i,j=0;for(i=0;iprod.size();i+)j=0;while(prod.get(i)+j!=-1);if(prod.get(i).length=j+1) /遇到空字符的动作continue;num=prod.get(i)j;prod.get(i)j=prod.get(i)j+1;prod.get(i)j+1=num;public void update()arraylist ary_n=new arraylist();arraylist ary_r;int index;int prod;for(int i=0;icore.size();i+)ary_n=core.get(i);ary_r=new arraylist();for(index=0;indexary_n.size();index+)prod=(int) ary_n.get(index);if(prod0=0|prod1!=-1)ary_r.add(prod);if(indexof(ary_c,ary_r)0)ary_c.add(ary_r);system.out.println();类:phrase功能:为每个项目配备向前搜索符输入:lr0核心项目集输出:lalr(1)核心项目集实现思想:lalr(1)的造核算法 public void sponsor()queue hq;int i,j;/核心项目lnodelinknode lnode;qnode qnode;/核心项目的产生式int p;/-2是一个特殊的标志符,也就是传播符串的标志;int s=-2;calfirst =new calfirst(producer);hq=new queue();/开始符号的特殊处理vt.add(#);i0.search=new int1;i0.search0=vt.indexof(#)+1000;qnode=new qnode(0,0);hq.enq(qnode);for(i=0;i0)for(j=0;j0)for(j=0;jitem.size();j+)qnode=update(item.get(j),lnode.search,hq);/修改传播搜索符并入队/if(qnode!=null)/hq.enq(qnode);/对每一个状态集都进行如此处理/求出项目p的闭包中所有核心项目的自生搜索符private int copy(int a )int val=new inta.length;for(int i=0;ia.length;i+)vali=ai;return val;public boolean closuer_self(int p,int s)int i;int left=-1;/产生式左部int temp;int genrate;int pos;int new_s;/arraylist pos=new arraylist();/找到-1(.)之后的第一个符号for(i=1;ip.length&pi!=-1;i+);i+;/p为移进项目if(i=1000)return false;/example:/0 1 -1 1 1003 2 /left = 1/s = first(1003,2,s);i+;/项目p中left之后的符号串加上s的first集就是left -1 xxx的搜索符集new_snew_s=first(p,i,s);pos=getstartwith(left);/closuer(p,pos);for(i=0;ipos.length;i+)genrate=producer.get(posi);/递归向下求解,暂时定为自生一次即截止,向下传播通过genarate的自生求解if(genrate.length=2)continue;if(!(left=p0)closuer_self(genrate,s);elseif(!isequal(genrate,p)closuer_self(genrate,s);/搜索符集是自生的if(new_s0!=-2)genrate=copy(genrate);temp=genrate1;genrate1=genrate2;genrate2=temp;item.add(genrate);sstr.add(new_s);return true;public void closuer(int item,arraylist pos)if(item.length=2)return ;int i;int p;for(i=0;iitem.length&itemi!=-1;i+);if(itemi+11000)p=getstartwith(itemi+1);for(i=0;ip.length;i+)if(pos.indexof(pi)=-1)pos.add(pi);closuer(producer.get(pi),pos);return ;/返回项目p所能传播到的所有项目public boolean closuer_spread(int p,int s)int i;int left=-1;/产生式左部符号int genrate;/int pos;int new_s;int temp;arraylist pos=new arraylist();/找到-1(.)之后的第一个符号for(i=1;ip.length&pi!=-1;i+);i+;/p为移进项目if(i=1000)return false;/example:/0 1 -1 1 1003 2 /left = 1/s = first(1003,2,s);i+;/项目p中left之后的符号串加上s的first集就是left -1 xxx的搜索符集new_sint temparr=new int1;temparr0=-2;new_s=first(p,i,temparr);temparr=null;/获得所有以left退出的项目/pos=getstartwith(left);this.closuer(p, pos);for(i=0;i=1个),并在第一个位置之后添加-1/本函数是建立在producer中的产生式的左部按升序基础上的,不可随意使用public int getstartwith(int i)int count=i;arraylist temp=new arraylist(3);while(counti)break;count+;int val=new inttemp.size();for(count=0;counttemp.size();count+)valcount=temp.get(count);return val;public int first(int prod,int i,int search)if(i=prod.length)return search;/calfirst.a.clear();/calfirst.firstv(prod, i, search);return calfirst.first(prod, i, search);private intarrcon(inta,int b)boolean ischange=false;arraylist temp=new arraylist();int i;if(a!=null)for(i=0;ia.length;i+)temp.add(ai);for(i=0;ib.length;i+)/有所更新if(temp.indexof(bi)=-1)temp.add(bi);ischange=true;a=new inttemp.size();for(i=0;ia.length;i+)ai=temp.get(i);if(ischange)return a;elsereturn null;public qnode update(int prod,int search,queue hq)/返回修改的项目是第几个状态集合中的(qnode.state)的第几个项目(qnode.item)(从0记数)linknode p;int a;int item;int i=0;while(ii.length)item=0;p=ii;while(p!=null)if(isequal(d,prod)a =arrcon(p.search,search);/null表示没有做任何修改if(a!=null)p.search =a;hq.enq( new qnode(i,item);return null;/此处是一个危险点,必须保证不能有一个项目在多个状态中出现,否则,会出现严重的错误item+;p=p.next;i+;return null;public boolean isequal(int a,int b)if(a.length !=b.length)return false;for(int i=0;ia.length ;i+)if(ai!=bi)return false;return true;类:analyze功能:整个分析程序的总控程序输入:编译目标文件、输出文件路径输出:语法分析树、三地址代码、错误信息实现思想:lr分析器工作原理class analyzeprivate arraylist producer =new arraylist(); private arraylist vt=new arraylist(); private arraylist vn=new arraylist(); / private linknode i; / 移进均为小于1000的,归约为大于1000的,并且接受状态的内容为-1,空状态用-2表示 private intaction; private intgoto;/goto private string prods; private arraylistlalr; defaultmutabletreenode root=null; public arraylist errorinfo=new arraylist(); public arraylist errorline=new arraylist(); public void analyze(string filepath,file outfile)fileinputstream f = null; try f = new fileinputstream(filepath); catch (filenotfoundexception e) e.printstacktrace(); /打开词法分析流: datainputstream input = new datainputstream(f); /lex为词法分析器 tinyclexer lex=new tinyclexer(input); try stack statestack =new stack(); /状态栈 stack tokenstack = new stack(); /符号栈的内容为:非终结符与vn中相对应,终结符-1000后与vt中相对应 int state; int temp; int prod; int i; int action; tokennode data; token token=null; token oldtoken=null; int tokentype; int end=vt.indexof(#)+1000;/#入栈 tree tree=new tree(); syntax syn=new syntax(tree,new filewriter(outfile); statestack.push(0);/状态0保证是开始状态 tokenstack.push(end); errorinfo.clear(); errorline.clear(); token=lex.nexttoken();/获得下一个面临的词法符 while(true) state=statestack.peek();/获得栈顶状态 if(token.gettype()=100)/无意义的跳过 continue; if(token.gettext()=null) token.settype(vt.indexof(#); /token.gettype()就是面临的输入符号(一定是终结符),它与vt中相对应 action=actionstatetoken.gettype(); if(action=-1) prod=producer.get(0); statestack.pop(); if(prod2!=tokenstack.pop() joptionpane.showmessagedialog(null, 编译程序遇到不能处理的错误!n分析被中止, 提示, joptionpane.information_message, new imageicon(imageserror.png); return ; tree.reduce(prod.length-2, vn.get(prod0); syn.syntaxanalyze(0); root= tree.getroot(); joptionpane.showmessagedialog(null, 编译程序完成编译!, 提示, joptionpane.information_message, new imageicon(imagesright.png); return ; if(actionstatetoken.gettype()1;i-)/产生式的右部的逆序为出栈顺序 statestack.pop(); if(prodi!=tokenstack.pop() joptionpane.showmessagedialog(null, 编译程序遇到不能处理的错误!n分析被中止, 提示, joptionpane.information_message, new imageicon(imageserror.png); return ; tree.reduce(prod.length-

温馨提示

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

评论

0/150

提交评论