编译原理课程设计lr语法分析构造器的设计_第1页
编译原理课程设计lr语法分析构造器的设计_第2页
编译原理课程设计lr语法分析构造器的设计_第3页
编译原理课程设计lr语法分析构造器的设计_第4页
编译原理课程设计lr语法分析构造器的设计_第5页
已阅读5页,还剩49页未读 继续免费阅读

下载本文档

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

文档简介

1、前言计算机语言之所以能由单一的机器语言发展到现今的数千种高级语言,就是因为有了编译技术,编译原理技术是计算机科学中发展的最迅速、最成熟的一个分支,它集中体现了计算机发展成果与精华。未来计算机工作者,都应该掌握这门基础的专业基础知识。“编译原理”是计算机及其相关专业的重要专业基础课,主要研究设计和构造编译程序的原理和方法。全面、深入地探讨了编译器设计方面的重要主题,包括词法分析、语法分析、语法制导定义和语法制导翻译、运行时刻环境、目标代码生成、代码优化技术、并行性检测以及过程间分析技。编译原理蕴涵着计算机学科中解决问题的思路、形式化问题和解决问题的方法,对应用软件和系统软件的设计与开发有一定的启

2、发和指导作用,编译程序构造的原理和技术在软件工程、语言转换等许多领域中有着广泛应用。语法分析是编译程序的核心部分。语法分析的作用是识别由词法分析给出的单词符号序列是否是给定文法的正确句子,目前语法分析常用的方法有自顶向下分析和自顶向上分析两大类。自顶向上分析包括确定分析和不确定分析,自顶向上分析又包括算符优先分析和LR分析。鉴于此,运用这些分析方法构造一个简单的分析程序是很有实践意义的。目 录编译原理课程设计任务书 .3第1章 概述.5 1.1 背景.5 1.2 目的.5 1.3 软件定义.51.4 开发环境.5第2章 需求分析.6 2.1 问题陈述.6 2.2 需完成的功能.6第3章 逻辑设

3、计. 7 3.1 模块设计.7 3.1.1 LR(1)项目集规范族的构造算法.8 3.1.2 LR(1)分析表的构造算法.8 3.2 流程图.9第4章 总体设计.15 4.1 构造项目集规范族模块.15 构造预测分析表模块.15 4.3 分析串程序模块.15第5章 界面设计.16小结.33致谢.34参考文献.35 附录 源程序清单.36编译原理课程设计任务书1、本课题的目的及意义课程设计实践对学生巩固所学基础专业课程知识、进行编译系统基本技能训练、培养实践动手能力,从而掌握编译系统的基本工作原理、基本方法和基本开发技术,最终达到具有一定的编译系统的实际开发能力有重要意义。通过课程设计,主要达到

4、以下目的:1.帮助学生深入理解编译原理的有关理论和巩固编译原理相关知识。2. 巩固学生学习的编译原理、程序设计语言、数据结构等课程的基础知识,训练学生分析和解决编译系统的相关问题的能力,提高学生的综合素质。3. 从软件工程的角度来看,编译原理课程设计是一个很好的实例,可以训练学生软件设计的能力以及编码调试能力。2、本课题任务的主要内容本课程设计主要内容包括以下几点:1、根据选定的题目,查阅资料,熟悉相关理论、方法;(1)掌握文献检索方法,以获得编译系统开发技术等相关资料;(2)学习并熟练使用一种4GL开发平台(如VC+、Java、Dephi、PB、VB等);2、分析问题,确定系统逻辑结构;3、

5、确定系统所需模块及模块结构,并用流程图描述各模块;4、编码及调试程序;5、撰写课程设计说明书。3、提交的成果1、一份符合课程设计说明书撰写规范的课程设计说明书。2、一套系统原型。附录一:课程设计说明书撰写要求1、基本要求:(1)能反映完成了设计内容要求;(2)要求撰写不少于5000个文字(20页)的文档;(3)文档中至少要包括:数据流图、逻辑结构图、系统功能图、算法流程图。(4)用户界面设计:附界面的抓图或手工绘图,及其主要核心部分代码。2、文档格式要求(参考课程设计参考模板)(1)封面(2)前言(3)目录(4)课程设计任务书(5)正文(分章、层次等,每一章从新一页开始)概述 包括项目背景、编

6、写目的、软件定义、开发环境等内容。需求分析 问题陈述、需完成的功能。以数据流图和数据字典表达。逻辑设计 描述系统组织和基本工作流程。以总体逻辑结构图表达。总体设计 画出软件功能图,描述每一个功能所完成的任务情况。界面设计 界面设计要合理,给出主要界面和主要代码并有适当的说明。(6)小结(7)参考文献对于引用的参考文献,列出主要参考文献(至少10篇)的题录及摘要或参考文献原文。(8)其他图表原始资料或参考资料附录第1章 概述 背景“编译原理”是计算机及其相关专业的重要专业基础课,主要研究设计和构造编译程序的原理和方法。全面、深入地探讨了编译器设计方面的重要主题,包括词法分析、语法分析、语法制导定

7、义和语法制导翻译、运行时刻环境、目标代码生成、代码优化技术、目标代码生成。语法分析是整个编译程序的核心部分,而LR分析方法对文法要求比起其他分析方法能力较强LR(k)分析方法是1965年Knuth提出的,括号中的k表示向右查看输入串符号的个数。这种方法比起自顶向下的LL(1)分析方法和自低向上的优先分析方法对文法的限制要少的多,也就是说对于大多数用无二义性上下文无关文法描述的语言都可以用相应的LR分析器进行识别,而且这种方法还具有分析速度快,能准确、即时地指出出错位置。自低向上分析的关键问题是在分析过程中如何确定句柄。LR分析法正是给出一种能根据当前分析栈中的符号串(通常以状态表示)和向右顺序

8、查看输入串的k个(k0)符号就可惟一地确定分析器的动作是移进还是归约和用哪个产生式归约,因而也就能惟一地确定句柄。LR分析法的归约过程是规范推到的逆过程,所以LR分析过程是一种规范归约的过程。因为LR(0)分析过程中不需要向右查看输入符号,因而它可以对文法的限制较大,对绝大多数的高级语言的语法分析器是不能适用的,所以,要分析绝大多数的高级语言编译程序的需要,采用向后查看一个输入符号的方法,即LR(1)的方法。(1)掌握并深刻理解有穷自动机在LR分析法中的应用(即LR分析器)。(2)掌握LR分析法的思想,学会特定分析表的构造方法,利用给出的分析表进行LR分析。1.3 软件定义对任意给定的上下文无

9、关文法G,构造其LR(1)项目集规范族、预测分析表,并且在此基础上进一步构造其LR(1)分析表。1.4 开发环境软件:Windows 7操作系统 , Microsof;编程语言:C+第2章 需求分析设计的题目是要对LR(1)类文法判定及其分析器进行构造。如果一个文法的LR(1)分析表不含多重入口时,(即任何一个LR(1)项目集中无移进归约冲突或归约归约冲突),则称该文法为LR(1)文法,所构造的相应分析表称为LR(1)分析,使用LR(1)分析表的分析器称为LR(1)分析器或称规范的LR分析器。所以,要判断一个文法是否是LR(1)类文法,则主要是看是否存在两个冲突。2.2 需完成的功能一个LR分

10、析器由3个部分组成:(1)总控程序:也可以成为驱动程序。对所有的LR分析器总控程序都是相同。(2)分析表或分析函数:不同的文法分析表将不同,同一个文法采用的LR分析器不同时,分析表也不同,分析表又可分为动作(ACTION)表和状态转换(GOTO)表两个部分,它们都可用二维数组表示。(3)分析栈:包括文法符号栈和相应的状态栈。它们均是先进后出栈。 分析器的动作由栈顶状态和当前输入符号所决定(LR(0)分析器不需要向前查看输入符号)。综上所述,要实现本次设计所需工作主要有以下方面:构造项目集规范族;构造预测分析表;设计总控程序,完成分析过程。第3章 逻辑设计3.1 模块设计 LR分析器工作过程示意

11、图如图1.1所示。其中SP为栈指针,Si为状态栈,Xi为文法符号栈。状态转换表内容按关系GOTOSi,X=Si确定,该关系式是指当栈顶状态为Si,遇到当前文法符号为X时应转向状态Sj。X为终结符或非终结符。 ACTIONSi,a规定了栈顶状态Si时遇到输入符号a应执行的动作。动作有4种可能:移进: 当Si=GOTOSi,a成立,则把Si移入到状态栈,把a移入到文法符号栈。其中i,j表示状态号。SPSnS1XnnX1X0S0. 总控程序ACTION表GOTO表输入串XXXXXXXX#输出图1-1 LR分析器工作过程示意图归约: 当在栈顶形成句柄为时,则用归约为相应的非终结符A,即当文法中有A的产

12、生式,而的长度为r(即|=r),则从状态和文法符号栈中自顶向下去掉r个符号,即栈指针SP减去r。并把A移入文法符号栈内,再把满足Sj=GOTIOSi,A的状态移入状态栈,其中Si为修改指针后的栈顶状态。接受acc: 当归约到文法符号栈中只剩文法的开始符号S时,并且输入符号串已结束即当输入是#,则为分析成功。报错: 当遇到状态栈顶为某一状态下出现不该遇到的文法符号时,则报错,说明输入串不是该文法所能接受的句子。LR分析器的关键部分是分析表的构造。构造LR分析表,那么先解决LR项目集规范族的构造。 LR项目集规范族的项目类型分为如下四种:移进项目圆点后为终结符的项目,形如Aa,其中、V*,aVT,

13、相应状态为移进状态。规约项目圆点在产生式右部的最后的项目,形如A其中V*,对于=的项目为A(对应的产生式为A),相应状态为归约状态。3、待约项目圆点后为非终结符的项目,形如AB,其中、V*,BVN,这表明用产生式A的右部归约时,首先要将B的产生式右部归约为B,对A的右部才能继续进行分析。也就是期待着继续分析过程中首先能进行归约得到B。4、接受项目当归约项目为SS时则表明已分析成功,即输入串为该文法的句子,相应状态为接受状态。.1 LR(1)项目集规范族的构造算法以SS,#属于初始项目集中,把“#”号作为向前搜索符,表示活前缀为(若是有关S产生式的某一右部)要归约成S时,必须面临输入符为“#”号

14、才行。我们对初始项目SS,#求闭包后再用转换函数逐步求出整个文法的LR(1)项目集族。具体构造步骤如下:构造LR(1)项目集的闭包函数。假定I是一个项目集,I的任何项目都属于CLOSURE(I)。AB,a属于CLOSURE(I),B是文法中的产生式,若有项目V*,bFIRST(a),则B,b也属于CLOSURE(I)中。重复直到CLOSURE(I)不再增大为止。构造转换函数。LR(1)转换函数的构造与LR(0)的相似,GO(I,X)= CLOSURE(J)其中I是LR(1)的项目集,X是文法符号:J=任何形如LR(1)分析表的构造AX,a的项目| AX,a I对文法G的LR(1)项目集族的构造

15、仍以SS,#为初态的初始项目,然后对其求闭包和转换函数,直到项目集不再增大。.2 LR(1)分析表的构造算法由于一个LR(1)项目可以看成两个部分组成,一部分和LR(0)项相同,这部分称为心,另一部分为向前搜索符合集。因而LR(1)分析表的构造与LR(0)分析表的构造在形式上基本相同,只是归约项目的归约动作取决于归约项目的向前搜索符集,即只有当面临的输入符属于向前搜索符的集合,才做归约动作,其他情况均出错。具体构造过程如下:若已构造出某文法的LR(1)项目集族C。C=I0,I1,.,In其中的Ik的k为分析器的状态,则动作ACTION表和状态转换GOTO表构造方法如下:(1) 若项目Aa,b属

16、于Ik,且GO(Ik,a)=Ij,其中aVT,则置ACTIONk,a=Sj。其Sj的含义是把输入符号a和状态j分别移入文法符号栈和状态栈。(2) 若项目A,a属于Ik,则置ACTIONk,a=rj,其中aVT,rj的含义为把当前栈顶符号串归约为A(即用产生式A归约)。j为在文法中对产生式A的编号。(3) 若项目SS,#属于Ik,则置ACTION,=“”,表示接受。(4) 若(Ik,),其中AVN ,则置GOTOk,A=j。表示转入j状态,则置当前文法符号栈顶为A,状态栈顶为j。(5) 凡不能用规则(1)(4)填入分析表中的元素,均置“报错标志”。可以填入空白以表示。 流程图图1-2(a) 构造

17、LR(1)项目集规范族流程图图1-2(b) 构造LR(1)项目集规范族流程图图1-2(c) 构造LR(1)项目集规范族流程图图1-3 构造LR(1)分析表流程图1-4 LR(1)串分析程序流程图第4章 总体设计功能设计为能实现本次课程设计的功能,同时减小工作量,根据需求,构造LR(1)项目集规范族需要用到FIRST集,主要思路是采用替换策略以及删除重复的元素。数据结构使用的是链表。这不是核心部分,所以不再赘述。将功能分为三个大模块,即构造项目集规范族、分析表、和分析串程序。以下是对构造项目集规范族、分析表、和分析串程序的设计叙述:4.1 构造项目集规范族模块项目集规范族是整个设计的基础。数据结

18、构:实现其构造,在数据结构选用方面要求能够方便查找、而且能够节省空间。据此,主要采用了C+中STL中队列、集合、向量,作为主要数据存放的容器,并用结构体存放重要产生式的重要信息。算法设计:由课本上关于构造项目集规范族的叙述,联想到了图的广度遍历,所以将扩展后的文法SS,#作为核放在队列中,由核求闭包,每求出一个就放入集合中(主要是可以去除重复的)如果插入集合成功,就把结构体插入向量中,插入队列中,直到队列空,说明由核产生的项目已经构造完,下一步只需从向量中取出项目进行闭包运算,直到把向量中所有元素取完,这样就可以把整个规范族求出。程序流程主要是一个WHILE循环和switch语句控制。4.2

19、构造预测分析表模块预测分析是分析过程的依据。数据结构:用的是结构体存放数据,数据信息主要包括:ACTION表,当前输入符、是移进还是归约的控制符号,归约的产生式编号和移进的状态。GOTO表,遇到的非终结符、转到的状态号。算法设计:这里主要是查找,利用上一步求出的项目集规范族,找出输入符号以后所转向的状态,或者是归约的产生式编号。4.3 分析串程序模块 分析串程序是最终功能的具体表示。数据结构:用的是两个先进先出栈,即状态栈和符号栈,另外还用一个string类型输入串。算法设计:根据预测分析表,开始时把0压入状态栈,把#压入符号栈中,根据状态栈的栈顶元素和输入串的串首字符两个线索,在分析表中查找

20、,找到相应的动作,并实现。如果找到,则继续分析,否则,说明该输入串是不能接受的。根据题目需要,还要判断是否是LR(1)文法,我们知道一个二义性文法绝不是LR类文法,对于这一功能也以与实现。主要的实现方法是通过分析表,如果分析表里有两个入口,也就是存在两个冲突,那么就说明不是。第5章 界面设计 图1-5 开始界面输入产生式文件代码如下:void main() void First(struct Vn pv2,int i,sq first,int c,char s2,char vn1); int Find(pseqstack S,sq st,int v); void dels(sq st,int

21、i);void project_set(char vn1,int c,struct set2 select,string ft); int Forecast_analyse(char vn1,int c);/进行预测分析 int i,n,t,h=0,j=0,c=0,l;char VN,T=; pl s1;string ft10; pseqstack S;sq first20,p;char vn120,string20,scanout20,s22=-,; FILE *fp; pv1=pv;pv4=pv2; S=Init_seqstack(); cout t * 说明 *n t *本系统设计了一些

22、产生式分别存放在 *n t * 以下文件中:1.txt,f.txt,H.txt *n t *in.txt,L.txt,L2.txt,ll.txt,lr.txt *n t *wd.txt结果输出存放在文件result.txt*n t *中,请用户按照上述文件名书写规范进*n t *行操作使用 *n t * 制作人:计算机071 李亚龙 *n t *欢迎使用*n t *=*n; printf(请输入所用产生式的文件名:); scanf(%s,scanout);if(fp=fopen(scanout,r)=NULL)printf(n打开文件出错了!n);exit(0);getchar(); s22=

23、0; cout 所用的文法是:endl; while(fgets(string,20,fp)!=NULL&!feof(fp)/从文件读入产生式 coutlength,s1 ); strinsert(pv2t.p, pv2t.p-length,s1 ); h+; vn1c=0; fclose(fp);/ 图1-6 构造出产生式项目集规范族代码如下:void project_set(char vn1,int c,struct set2 select,string ft)queuequ;/队列Item product_Item70;/小集合容器setItem_set;/大集合vectorvt; co

24、ntainer pi,pi3,pi150,product_It;vectorpi2;int i,j,k,n,n1,z=1,v=-1,l=0,q=0,q1=0,flat=0;static f=0;char ch;bool bl;string str,str1,str2,str3;Item_container product1,product2,product3; fstream of;product1.s=.S;product1.s2=0;product1.pcode=0;product1.front=#;product1.go=S;product1.point_pos=0;product1.v

25、n=S;product_Itemq.insert (product1.vn+-+product1.s+,+product1.front);qu.push(product1); pi.push_back(product1); pi1q1.push_back(product1);while(flat!=100) switch(flat)case 0:if(!qu.empty() product2=qu.front();/队首元素 qu.pop();/删除队首元素 if(isVN(vn1,product2.go,c)/判断点后是否为非终结符 k=0; for(i=0,n=0;product2.spr

26、oduct2.point_pos+1!=vn1i;n+=pv2i.tag,i+);/找到非终结符的下标 k=pv2i.tag;/该非终结符的产生式个数n1=n;for(j=0;j+product1.s+,+product1.front).second=true)qu.push(product1);pi.push_back(product1);pi1q1.push_back(product1); flat=0; else flat=1; break;case 1: bl=Item_set.insert(product_Itemq).second; if(bl=true) f+; pi2.push

27、_back(pi1q1); if(!vt.empty()flat=3;elseif(vq|v=f)/flat=100;else flat=2; v+;break; case 2: if(vq|v=f)flat=100;else i=0; for(vector:iterator it=pi2.begin();iv;it+,i+); product_It=*it; for(vector:iterator vi=product_It.begin();vi!=product_It.end();vi+) vt.push_back(*vi).vn+-+(*vi).s+,+(*vi).front); fla

28、t=3; break;case 3: str=vt.back(); vt.pop_back(); flat=4; break; case 4:for(vector:iterator vc=pi.begin();vc!=pi.end();vc+)/找到str对应的产生式 product3=*vc; if(product3.vn+-+product3.s+,+product3.front)=str&product3.go!=$&product3.go!=) ch=product3.go;place(product3.point_pos,1,1,product3.go); product3.poin

29、t_pos+=1; str2.replace(product3.point_pos,1,1,.); product3.s=str2; if(product3.sproduct3.point_pos+1=0) product3.go=$; else product3.go=product3.sproduct3.point_pos+1; qu.push(product3); pi.push_back(product3); product_Item+q.insert(product3.vn+-+product3.s+,+product3.front); pi1+q1.push_back(produc

30、t3);z=0; break; if(z=0)if(ch!=$)for(vector:iterator vv=vt.begin();vv!=vt.end();vv+)/找到相同输入符str1=*vv; for(vector:iterator vc1=pi.begin();vc1!=pi.end();vc1+) product1=*vc1; if(product1.go!=$)if(product1.go=ch&(product1.vn+-+product1.s+,+product1.front)=str1) str3=product1.vn+-+product1.s+,+product1.fr

31、ont; str2=product1.s; str2.replace(product1.point_pos,1,1,product1.go); product1.point_pos+=1; str2.replace(product1.point_pos,1,1,.); product1.s=str2;if(product1.sproduct1.point_pos+1!=0)product1.go=product1.sproduct1.point_pos+1;else product1.go=$;product_Itemq.insert(product1.vn+-+product1.s+,+pr

32、oduct1.front); qu.push(product1); pi.push_back(product1); pi1q1.push_back(product1);vector:iterator i=find(vt.begin(),vt.end(),str3); if(i!=vt.end()vt.erase(i); vv-; break;else vt.pop_back();l=1; break; z=1; if(l=1) l=0; break; flat=0; break; of.open(result.txt,ios:out);if(!of)cerr打开文件失败!endl;abort(

33、);cout 构造的项目集规范族是:(按回车键继续)endl;getchar();z=0;for(vector:iterator vc2=pi2.begin();vc2!=pi2.end();vc2+) pi=*vc2;coutIz:endl;ofIz:endl; z+;for(vector:iterator pl=pi.begin();pl!=pi.end();pl+)cout(*pl).vn(*pl).s,(*pl).frontendl;of(*pl).vn(*pl).s,(*pl).frontendl;getchar();coutendl;ofendl;of.close();/*构造预测

34、分析表*/ i=-1;for(vector:iterator vc3=pi2.begin();vc3!=pi2.end();vc3+) pi=*vc3; i+;j=0; int t=-1; for(vector:iterator pl=pi.begin();pl!=pi.end();pl+) product1=*pl;if(product1.vn+-+product1.s=S-S.)tablei.actj.ch=#; tablei.actj.control=a; tablei.actj.data=0;j+;else if(product1.vn+-+product1.s=product1.vn

35、+-+.) for(k=0;product1.frontk!=0;k+) tablei.actj.ch=product1.frontk; tablei.actj.control=r; tablei.actj.data=product1.pcode; j+; else if(product1.point_pos=selectproduct1.pcode.len) for(k=0;product1.frontk!=0;k+) tablei.actj.ch=product1.frontk; tablei.actj.control=r; tablei.actj.data=product1.pcode;

36、 j+; else if(!isVN(vn1,product1.sproduct1.point_pos+1,c) tablei.actj.ch=product1.go; tablei.actj.control=s; else tablei.to+t.cha=product1.go; str2=product1.s; str2.replace(product1.point_pos,1,1,product1.go); str2.replace(product1.point_pos+1,1,1,.); str3=product1.vn+-+str2+,+product1.front; l=0; fo

37、r(vector:iterator vc4=pi2.begin();vc4!=pi2.end();vc4+,l+) pi3=*vc4; for(vector:iterator ppl=pi3.begin();ppl!=pi3.end();ppl+) if(str3=(*ppl).vn+-+(*ppl).s+,+(*ppl).front) if(!isVN(vn1,product1.sproduct1.point_pos+1,c) tablei.actj.data=l; else tablei.tot.da=l; n=-1; break; if(n=-1) n=0; break; j+; tab

38、lei.a=j; tablei.t=t;/预测分析表 图1-7 输入串分析接受的情况 图1-8 输入串分析不接受的情况代码如下: int Forecast_analyse(char vn1,int c)/进行预测分析 FILE* op;stackstate;stacksymbol;char input20,stack20,remain20,remain120,stack120;int i,j=0,l,b=0,z,r=0,n=0,s,ss,k;int sta20,sta120;char x;if(op=fopen(result.txt,a)=NULL) printf(n打开文件出错了!n); e

39、xit(0); printf(请输入要分析的串,以#结束:n);gets(input); printf( 输入串%s的分析过程 n,input); printf(-n);fprintf(op, 输入串%s的分析过程 n,input); fprintf(op,-n);for(z=0;inputz!=0;z+)remainz=inputz; remainz=0; state.push(0);symbol.push(#); stack0=#;stack1=0;sta0=0; out(步骤, 0,5); out(状态栈,0,10); out(符号栈,0,10);out(输入串,2,10);out(AC

40、TION,2,8);out(GOTO,2,6);coutendl;fprintf(op,步骤 状态栈 符号栈 输入串 ACTION GOTOn);do b+; s=state.top();x=inputn; for(i=0;i=tables.a;i+) if(tables.acti.control=a) out(b, 0,5);fprintf(op,%-7d,b); out(sta0,0,0);fprintf(op,%-d,sta0); for(k=1;stackk+1!=0;k+) out(stak,0,0);fprintf(op,%-d,stak); out(stak,0,10-k);fp

41、rintf(op,%-8d,stak); out(stack,0,10); fprintf(op,%-10s,stack); out(remain,2,10);fprintf(op,%+10s,remain);out(接受,2,8);fprintf(op, 接受n );coutendl; cout分析成功endl;fprintf(op, 分析成功n ); return 0; else if(tables.acti.ch=x) if(tables.acti.control=s) state.push(tables.acti.data); symbol.push(x); n+; if(b=1) o

42、ut(b, 0,5); fprintf(op,%-7d,b); out(0,0,10);fprintf(op,%-8d,sta0); out(stack,0,10);fprintf(op,%-10s,stack);out(remain,2,10);fprintf(op,%+10s,remain);out(s,2,6);fprintf(op, s);out(tables.acti.data,0,0);fprintf(op,%-dn,tables.acti.data);cout1) out(b, 0,5);fprintf(op,%-7d,b); out(sta10,0,0); fprintf(op

43、,%-d,sta10); for(k=1;stack1k+1!=0;k+) out(sta1k,0,0);fprintf(op,%-d,sta1k); out(sta1k,0,10-k);fprintf(op,%-8d,sta1k); out(stack1,0,10);fprintf(op,%-10s,stack1); out(remain1,2,10);fprintf(op,%+10s,remain1);out(s,2,6);fprintf(op, s);out(tables.acti.data,0,0);fprintf(op,%-dn,tables.acti.data);coutendl;

44、 break; else if(tables.acti.control=r) for(k=0;stackk!=0;k+)stack1k=stackk; sta1k=stak; stack1k=0;l=tables.acti.data;for(k=0;stackk!=0;k+); if(selectl.NVT0=) ss=state.top(); for(i=0;itabless.t&tabless.toi.cha!=vn1selectl.n;i+); state.push(tabless.toi.da); symbol.push(vn1selectl.n); stak-j=tabless.to

45、i.da; stackk-j=vn1selectl.n; stackk-j+1=0; else for(j=0;jselectl.len;j+) state.pop(); symbol.pop(); stackk-j=0; ss=state.top(); for(i=0;itabless.t&tabless.toi.cha!=vn1selectl.n;i+); state.push(tabless.toi.da); symbol.push(vn1selectl.n); stak-j=tabless.toi.da; stackk-j=vn1selectl.n; stackk-j+1=0; if(

46、b=1) out(b, 0,5);fprintf(op,%-7d,b); out(0,0,10);fprintf(op,%-8d,sta0); out(stack,0,10);fprintf(op,%-10s,stack); out(remain,2,10);fprintf(op,%+10s,remain);out(r,2,6);fprintf(op, r);out(tables.acti.data,0,0);fprintf(op,%-5d,tables.acti.data);out(tabless.toi.da,2,6);fprintf(op,%3dn,tabless.toi.da);cou

47、t1) out(b, 0,5);fprintf(op,%-7d,b); out(sta10,0,0);fprintf(op,%-d,sta10); for(k=1;stack1k+1!=0;k+) out(sta1k,0,0);fprintf(op,%-d,sta1k); out(sta1k,0,10-k);fprintf(op,%-8d,sta1k); out(stack1,0,10);fprintf(op,%-10s,stack1); out(remain,2,10);fprintf(op,%+10s,remain);out(r,2,6);fprintf(op, r);out(tables

48、.acti.data,0,0);fprintf(op,%-5d,tables.acti.data);out(tabless.toi.da,2,6);fprintf(op,%3dn,tabless.toi.da);coutendl; if(tables.act0.control=a) out(b, 0,5);fprintf(op,%-7d,b); out(sta10,0,0);fprintf(op,%-d,sta10); for(k=1;stack1k+1!=0;k+) out(sta1k,0,0);fprintf(op,%-d,sta1k); out(sta1k,0,10-k);fprintf

49、(op,%-8d,sta1k); out(stack1,0,10);fprintf(op,%-10s,stack1); out(remain,2,10);fprintf(op,%+10s,remain);out(接受,2,8);fprintf(op, 接受n );couttables.a) out(b, 0,5);fprintf(op,%-7d,b); out(sta0,0,0);fprintf(op,%-d,sta10); for(k=1;stackk+1!=0;k+) out(stak,0,0);fprintf(op,%-d,stak); out(stak,0,10-k);fprintf(

50、op,%-8d,stak); out(stack,0,10);fprintf(op,%-10s,stack); out(remain,2,10);fprintf(op,%+10s,remain);out(不能接受,2,10);fprintf(op, 不能接受n );coutendl; return -1; while(1);图1-9 对于不是LR(1)文法的判断代码如下:for(i=0;iz;i+) for(j=0;jtablei.a;j+) for(l=j+1;l=tablei.a;l+) if(tablei.actj.ch=tablei.actl.ch&tablei.actj.contro

51、l!=tablei.actl.control)| (tablei.actj.ch=tablei.actl.ch&tablei.actj.control=r&tablei.actl.control=r& tablei.actj.data!=tablei.actl.data) cout不是LR(1)文法endl; cout不能继续分析,退出程序!endl; exit(0); 小 结我们班的题目是LR(1)类文法的判定及其分析器的构造,要实现这些功能,可谓是绞尽脑汁。实现过程并不是顺利的,由不了解到实现,最终还是做到了。对于本次的课程设计,可以说是我做过课程设计以来,收获最大的一个。我之所以这么说主

52、要是因为,从设计的难度方面,是比较大的。到具体的实现,其中要考虑很多细节问题,比如说,输入符是否是终结符,产生式含空的情况等一些小问题,影响到了设计的进度和功能的完整。如何能方便的进行设计,降低难度,查阅资料,现学一些东西,比如,用C+中标准模板库STL,还有模板降低难度。当程序遇到问题,最难得是调试的过程,尤其构造项目集规范族时,进行调试,花了很长时间,因为对内部的一些东西不了解,增加了调试难度。在这过程中,也曾经气馁过,但是最后还是坚持了下来。其实刚开始时,是不觉的难,到中间阶段感到很难。不过现在回头再看的时候觉的也不是想象的复杂。通过这次课程设计,我学到了很多东西,收获了很多,对于LR(

53、1)分析有了更深的理解,对C+中泛型编程的实现也有了一些心得体会。在成功的同时我也发现自己的一些不足。知识的不足,课本上知识掌握的不是很牢。编程能力还很欠缺,思维不是很清晰。缺乏耐心。还有就是做出的程序还不够完美,界面不是很美观。总之,虽然实现了最终结果,但是我对自己并不满意。还有很多等待完善的地方,当然不仅仅是程序,更重要的是完善自己,提高自己的能力。致 谢在这次课程设计的撰写过程中,我得到了许多人的帮助。首先我要感谢我的父母,是他们养育了我,父母的爱伴随着我的成长,每当我遇到挫折和失败的时候,父母总是能够给我最大的安慰和帮助,有了他们我才能成长,才能进入大学这个环境;其次我要感谢我所在的大

54、学安徽工程大学,感谢学校为我们提供了这样好的学习环境,再次我要感谢我的老师在课程设计上给予我的指导、提供给我的支持和帮助,这是我能顺利完成这次报告的主要原因,在此期间,我不仅学到了许多新的知识,而且也开阔了视野,提高了自己的设计能力。最后,我要感谢帮助过我的同学,他们也为我解决了不少我不太明白的设计上的难题。 李亚龙 2010 年6月10日参考文献【1】张素琴、吕映芝等.编译原理(第二版).清华大学出版社 , 2008年【2】美scott Meyers 著 潘爱民译. Effective STL中文版 .清华大学出版社 ,2006年【3】陈本林 著 .数据结构使用C+标准模板库(STL). 机

55、械工业出版社, 2005年【4】MENGLEE著 王昕译 .C+ STL中文版. 中国电力出版社 ,2002年【5】陈火旺.程序设计语言编译原理(第三版).国防工业出版社,2000年【6】高伸仪 金茂忠.编译原理及编译程序构造.北京航空航天大学出版社,2004年【7】郑莉 董渊.C+语言程序设计.清华大学出版社,2006年【8】钱能.C+程序设计教程.清华大学出版社,2006年【9】William Ford,William Topp 编著,刘卫东等译.数据结构C+语言描述.清华大学出版社,2003年 【10】王雷.编译原理课程设计.机械工业出版社,2006年附 录源程序清单#pragma wa

56、rning(disable:4786)#pragma warning(disable:4503)#include#include#include#include#include#include#include#include#include#include#includeusing namespace std;#define MAX 20#define MAXSIZE 100 typedef struct productstring s;int pcode;string front;char go;int point_pos;string vn;Item_container;typedef s

57、etItem;typedef vectorcontainer;typedef struct chuan/定义一个堆串 char *ch; int length; l,*pl;struct Vn/*产生式结构*/int tag;/每个非终结符的产生式个数int info;/标记能否推出空char ch;/标记非终结符 pl p;pv20,pv220,*pv1,*pv4;struct set2/定义SELECT集int n;/非终结符的下标int len;/产生式的长度char NVT10;/存放产生式的右部select20,*ss;typedef struct node/*first和follo

58、w集合结构*/int lable;/*表明是否是非终结符*/int f;/*表明是first还是 follow集*/unionint key;/终结符的下标char ch;/终结符的字符val;struct node *next;*sq,sql;typedef structint t,v;info;typedef struct/*定义栈结构,递归用*/ info dataMAX; int top; seqstack,*pseqstack;typedef struct/*定义栈结构,递归用*/ char tittleMAXSIZE; int top; sign,*psign;typedef st

59、ructchar ch;/遇到的字符char control;/是否归约还是已经int data;/移进的状态号或者归约的产生式action;typedef structchar cha;/遇到的非终结符int da;/状态号go;struct analysisaction act30;go to20;int a;int t;table50;static int visited20;/标记数组static int y;template void out(T1 t1,T2 t2,T2 t3)if(t2=0)cout.width(t3);cout.setf(ios:left,ios:adjustf

60、ield);coutsetfill( )t1;elseif(t2=1)cout.width(t3);cout.setf(ios:internal,ios:adjustfield);coutsetfill( )t1;elsecout.width(t3);cout.setf(ios:right,ios:adjustfield);coutsetfill( )top=-1; return s;int push_seqstack(pseqstack s,info temp)if(s-top=MAX-1) return 0; else s-top+; s-datas-top.t=temp.t; s-dat

温馨提示

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

评论

0/150

提交评论