ImageVerifierCode 换一换
格式:DOCX , 页数:38 ,大小:86.62KB ,
资源ID:30183738      下载积分:3 金币
快捷下载
登录下载
邮箱/手机:
温馨提示:
快捷下载时,用户名和密码都是您填写的邮箱或者手机号,方便查询和重复下载(系统自动生成)。 如填写123,账号就是123,密码也是123。
特别说明:
请自助下载,系统不会自动发送文件的哦; 如果您已付费,想二次下载,请登录后访问:我的下载记录
支付方式: 支付宝    微信支付   
验证码:   换一换

加入VIP,免费下载
 

温馨提示:由于个人手机设置不同,如果发现不能下载,请复制以下地址【https://www.bdocx.com/down/30183738.html】到电脑端继续下载(重复下载不扣费)。

已注册用户请登录:
账号:
密码:
验证码:   换一换
  忘记密码?
三方登录: 微信登录   QQ登录  

下载须知

1: 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。
2: 试题试卷类文档,如果标题没有明确说明有答案则都视为没有答案,请知晓。
3: 文件的所有权益归上传用户所有。
4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
5. 本站仅提供交流平台,并不能对任何下载内容负责。
6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

版权提示 | 免责声明

本文(语法分析器实验报告.docx)为本站会员(b****5)主动上传,冰豆网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对上载内容本身不做任何修改或编辑。 若此文所含内容侵犯了您的版权或隐私,请立即通知冰豆网(发送邮件至service@bdocx.com或直接QQ联系客服),我们立即给予删除!

语法分析器实验报告.docx

1、语法分析器实验报告语法分析器的设计实验报告一、实验内容语法分析程序用LL(1)语法分析方法。首先输入定义好的文法书写文件(所用的文法可以用LL(1)分析),先求出所输入的文法的每个非终结符是否能推出空,再分别计算非终结符号的FIRST集合,每个非终结符号的FOLLOW集合,以及每个规则的SELECT集合,并判断任意一个非终结符号的任意两个规则的SELECT集的交集是不是都为空,如果是,则输入文法符合LL(1)文法,可以进行分析。对于文法:GE:E-E+T|TT-T*F|FF-i|(E)分析句子i+i*i是否符合文法。二、基本思想1、语法分析器实现语法分析是编译过程的核心部分,它的主要任务是按照

2、程序的语法规则,从由词法分析输出的源程序符号串中识别出各类语法成分,同时进行词法检查,为语义分析和代码生成作准备。这里采用自顶向下的LL(1)分析方法。语法分析程序的流程图如图5-4所示。该程序可分为如下几步:(1)读入文法 (2)判断正误 (3)若无误,判断是否为LL(1)文法 (4)若是,构造分析表;(5)由句型判别算法判断输入符号串是为该文法的句型。三、核心思想该分析程序有15部分组成:(1)首先定义各种需要用到的常量和变量;(2)判断一个字符是否在指定字符串中;(3)读入一个文法;(4)将单个符号或符号串并入另一符号串;(5)求所有能直接推出&的符号;(6)求某一符号能否推出 & ;(

3、7)判断读入的文法是否正确;(8)求单个符号的FIRST;(9)求各产生式右部的FIRST;(10)求各产生式左部的FOLLOW;(11)判断读入文法是否为一个LL(1)文法;(12)构造分析表M;(13)句型判别算法;(14)一个用户调用函数;(15)主函数;下面是其中几部分程序段的算法思想:1、求能推出空的非终结符集、实例中求直接推出空的empty集的算法描述如下:void emp(char c) 参数c为空符号 char temp10;定义临时数组 int i; for(i=0;iB,B可推出空) if 右部长度为1但第一个字符为终结符,then 返回0(A-a,a为终结符) else

4、for(k=0;kAB) if 找到的字符与当前字符相同(A-AB) 结束本次循环 else(mark等于0)查找右部符号是否可推出空字,把返回值赋给result 把当前符号加入到临时数组empt里. if 当前字符不能推出空字且还没搜索完全部的产生式then 跳出本次循环继续搜索下一条产生式 else if /当前字符可推出空字,返回1 2、计算每个符号的first集:实例中求单个符号的FIRST集的算法描述如下:void first2 (int i) 参数i为符号在所有输入符号中的序号 c等于指示器i所指向的符号 在保存终结符元素的termin数组查找cif c为终结符(cVT ),the

5、n FIRST(c)=c在保存终结符元素的non_ter数组查找cif c是非终结符(cVN )在所有产生式中查找c所在的产生式if 产生式右部第一个字符为终结符或空(即ca (aVT)或c&) then把a或&加进FIRST(c)if 产生式右部第一个字符为非终结符 thenif 产生式右部的第一个符号等于当前字符 then 跳到下一条产生式进行查找求当前非终结符在所有字符集中的位置if 当前非终结符还没求其FIRST集 then 查找它的FIRST集并标识此符号已求其FIRST集 求得结果并入到c的FIRST集.if 当前产生式右部符号可推出空字且当前字符不是右部的最后一个字符 then获

6、取右部符号下一个字符在所有字符集中的位置if 此字符的FIRST集还未查找 then找其FIRST集,并标其查找状态为1把求得的FIRST集并入到c的FIRST集.if当前右部符号串可推出空且是右部符号串的最后一个字符(即产生式为cY1Y2Yk,若对一切1=i=k,均有&FIRST(Yi),则将&符号加进FIRST(c) ) then把空字加入到当前字符c的FIRST集. else不能推出空字则结束循环标识当前字符c已查找其FIRST集. 3. 计算FOLLOW集FOLLOW集的构造可用如下方法来求:对于文法中的符号X VN ,其FOLLOW(A)集合可反复应用下列规则计算,直到FOLLOW(

7、A)集合不再增大为止。(1)对于文法开始符号S,因为S S,故# FOLLOW(S);(2)若A B,其中B VN, (VT VN)*、 (VT VN)+,则FIRST()- FOLLOW(B);(3)若A B或A B ( ),则FOLLOW(A) FOLLOW(B)。 FOLLOW集的算法描述如下: void FOLLOW(int i) X为待求的非终结符把当前字符放到一临时数组foll中,标识求已求其FOLLOW集.避免循环递归if X为开始符号 then #FOLLOW(X) 对全部的产生式找一个右部含有当前字符X的产生式注:比如求FOLLOW(B)则找AX或AX()的产生式if X在产

8、生式右部的最后(形如产生式AX) then查找非终结符A是否已经求过其FOLLOW集.避免循环递归if 非终结符A已求过其FOLLOW集 thenFOLLOW(A)FOLLOW(X)继续查下一条产生式是否含有Xelse求A之FOLLOW集,并标识为A已求其FOLLOW集else if X不在产生式右部的最后(形如AB) thenif右部X后面的符号串能推出空字 then查找是否已经求过其FOLLOW集.避免循环递归if 已求过的FOLLOW集 then FOLLOW(A)FOLLOW(B)结束本次循环else if 不能推出空字 then 求FIRST()把FIRST()中所有非空元素加入到F

9、OLLOW(B)中标识当前要求的非终结符X的FOLLOW集已求过4.计算SELECT集SELECT集的构造算法如下:对所有的规则产生式Ax:(1)若x不能推出空字,则SELECT(Ax) = FIRST(x);(2)若x可推出空字,则SELECT(Ax)=FIRST(x) FOLLOW(A)。算法描述如下: for(i=0;i=产生式总数-1;i+)先把当前产生式右部的FIRST集(一切非空元素,不包括)放入到当前产生式的SELECT(i); if 产生式右部符号串可推出空字 then把i指向的当前产生式左部的非终结符号的FOLLOW集并入到SELECT(i)中5.判断是否LL(1)文法要判断

10、是否为LL(1)文法,需要输入的文法G有如下要求:具有相同左部的规则的SELECT集两两不相交,即:SELECT(A) SELECT(A)= 如果输入的文法都符合以上的要求,则该文法可以用LL(1)方法分析。算法描述如下: 把第一条产生式的SELECT(0)集放到一个临时数组temp中 for(i=1;i=产生式总数-1;i+) 求temp的长度length if i指向的当前产生式的左部等于上一条产生式的左部 then 把SELECT(i)并入到temp数组中 If temp的长度小于length加上SELECT (i)的长度 返回0 else把temp清空把SELECT (i)存放到tem

11、p中结果返回1;四、算法#include#include#include/*/int count=0; /产生式的个数int number; /所有终结符和非终结符的总数char start; /开始符号char termin50; /终结符号char non_ter50; /非终结符号char v50; /所有符号char left50; /左部char right5050; /右部char first5050,follow5050; /各产生式右部的FIRST和左部的FOLLOW集合char first15050; /所有单个符号的FIRST集合char select5050; /各个产生

12、式的SELECT集合char firstflag50,followflag50; /记录各符号的FIRST和FOLLOW是否已求过char empty20; /记录可推出&的符号char nonempty20; /记录不可推出&的符号char empt20; /求_emp()时使用char TEMP50; /求FOLLOW时存放某一符号串的FIRST集合int validity=1; /表示输入文法是否有效int ll=1; /表示输入文法是否为LL(1)文法int M2020; /分析表char choose; /用户输入时使用char foll20; /求FOLLOW集合时使用/*判断一个

13、字符c是否在指定字符串p中*/int in(char c,char *p) int i; if(strlen(p)=0) return(0); for(i=0;i+) if(pi=c) return(1); /若在,返回1 if(i=(int)strlen(p) return(0); /若不在,返回0 /*将单个符号或符号串并入另一符号串*/void merge(char *d,char *s,int type) /是目标符号串,s是源串,type1,源串中的&一并并入目串; /type2,源串中的&不并入目串 int i,j; for(i=0;i=(int)strlen(s)-1;i+) i

14、f(type=2&si=&); else for(j=0;j+) if(j(int)strlen(d)&si=dj) break; /若已存在,则退出,继续看下一个源串字符 if(j=(int)strlen(d) /若不存在,则并入 dj=si; dj+1=0; break; /*读入一个文法*/char grammer(char *t,char *n,char *left,char right5050) char vn50,vt50; char s; char p5050; int i,j; printf(请输入文法的非终结符号串:); scanf(%s,vn); getchar(); i=

15、strlen(vn); memcpy(n,vn,i); ni=0; printf(请输入文法的终结符号串:); scanf(%s,vt); getchar(); i=strlen(vt); memcpy(t,vt,i); ti=0; printf(请输入文法的开始符号:); scanf(%c,&s); getchar(); printf(请输入文法产生式的条数:); scanf(%d,&i); getchar(); count=i; for(j=1;j=i;j+) printf(请输入文法的第%d条(共%d条)产生式:,j,i); scanf(%s,pj-1); getchar(); for(

16、j=0;j) /检测输入错误 printf(n输入错误!); validity=0; return(0); return(s);/*判断读入的文法是否正确*/int judge() int i,j; for(i=0;i=count-1;i+) if(in(lefti,non_ter)=0) /若左部不在非终结符中,报错 printf(n文法左部出错!); validity=0; return(0); for(j=0;j=(int)strlen(righti)-1;j+) if(in(rightij,non_ter)=0&in(rightij,termin)=0&rightij!=&) /若右部

17、某一符号不在非终结符、终结符中且不为&,报错 printf(n文法右部出错!); validity=0; return(0); return(1);/*求所有能直接推出&的符号*/void emp(char c) char temp10; int i; for(i=0;iB,B可推出空) return(1); else if(j=1&in(righti0,termin)=1)/右部长度为1但第一个字符为终结符,返回0(A-a,a为终结符) continue; else for(k=0;kAB) mark=1; if(mark=1) /找到的字符与当前字符相同(A-AB) continue; /

18、结束本次循环 else /(mark等于0) for(k=0;k=j-1;k+) result*=_emp(rightik);/递归调用,查找右部符号是否可推出空字,把返回值赋给result temp0=rightik; temp1=0; merge(empt,temp,1);/把当前符号加入到临时数组empt里,标记已查找 if(result=0&icount)/如果当前字符不能推出空字且还没搜索完全部的产生式,则跳出本次循环继续搜索下一条产生式 continue; else if(result=1&icount)/当前字符可推出空字,返回1 return(1); /*求单个符号的FIRST

19、*/void first2(int i) /i为符号在所有输入符号中的序号 char c,temp20; int j,k,m; char ch=&; c=vi; emp(ch);/求所有能直接推出空字的符号,结果保存到empty中 if(in(c,termin)=1) /若为终结符-cVT,则FIRST(c)=c first1i0=c; first1i1=0; else if(in(c,non_ter)=1) /若为非终结符 for(j=0;j=count-1;j+) /j为所有产生式中的序列 if(leftj=c) /找一个左部为c的产生式 if(in(rightj0,termin)=1|r

20、ightj0=&) /若产生式右部第一个字符为终结符或空.-产生式Xa (aVT)或X&,则把a或&加进FIRST(X) temp0=rightj0; temp1=0; merge(first1i,temp,1); /-XY1Y2Yk的产生式,若Y1VN,则把FIRST(Y1)中的一切非空符号加进FIRST(X) else if(in(rightj0,non_ter)=1)/产生式右部第一个字符为非终结符 if(rightj0=c)/产生式右部的第一个符号等于当前字符,则跳到下一条产生式进行查找 continue; for(k=0;k+) if(vk=rightj0)/求右部第一个字符在所有字

21、符集中的位置k break; if(firstflagk=0) first2(k);/求其FIRST集 firstflagk=1;/标识其为查找状态 merge(first1i,first1k,2);/求得结果并入到X的FIRST集. for(k=0;k(int)strlen(rightj);k+) empt0=0;/存放到一个临时数组里,标识此字符已查找其是否可推出空字 if(_emp(rightjk)=1&k(int)strlen(rightj)-1) /当前产生式右部符号可推出空字,且当前字符不是右部的最后一个字符 for(m=0;m+) if(vm=rightjk+1)/获取右部符号下一个字符在所有字符集中的位置 break;

copyright@ 2008-2022 冰豆网网站版权所有

经营许可证编号:鄂ICP备2022015515号-1