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

加入VIP,免费下载
 

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

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

下载须知

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

版权提示 | 免责声明

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

语法分析器实验报告Word格式.docx

1、1、求能推出空的非终结符集、实例中求直接推出空的empty集的算法描述如下:void emp(char c) 参数c为空符号 char temp10;定义临时数组 int i; for(i=0;iB,B可推出空) if 右部长度为1但第一个字符为终结符,then 返回0(A-a,a为终结符) else for(k=0;kAB) if 找到的字符与当前字符相同(A- 结束本次循环 else(mark等于0)查找右部符号是否可推出空字,把返回值赋给result 把当前符号加入到临时数组empt里. if 当前字符不能推出空字且还没搜索完全部的产生式then 跳出本次循环继续搜索下一条产生式 els

2、e if /当前字符可推出空字,返回1 2、计算每个符号的first集:实例中求单个符号的FIRST集的算法描述如下:void first2 (int i) 参数i为符号在所有输入符号中的序号 c等于指示器i所指向的符号 在保存终结符元素的termin数组查找cif c为终结符(cVT ),then FIRST(c)=c在保存终结符元素的non_ter数组查找cif c是非终结符(cVN )在所有产生式中查找c所在的产生式if 产生式右部第一个字符为终结符或空(即ca (aVT)或c&) then把a或&加进FIRST(c)if 产生式右部第一个字符为非终结符 thenif 产生式右部的第一个

3、符号等于当前字符 then 跳到下一条产生式进行查找求当前非终结符在所有字符集中的位置if 当前非终结符还没求其FIRST集 then 查找它的FIRST集并标识此符号已求其FIRST集 求得结果并入到c的FIRST集.if 当前产生式右部符号可推出空字且当前字符不是右部的最后一个字符 then获取右部符号下一个字符在所有字符集中的位置if 此字符的FIRST集还未查找 then找其FIRST集,并标其查找状态为1把求得的FIRST集并入到c的FIRST集.if当前右部符号串可推出空且是右部符号串的最后一个字符(即产生式为cY1Y2Yk,若对一切1=i=k,均有&FIRST(Yi),则将&符号

4、加进FIRST(c) ) then把空字加入到当前字符c的FIRST集. else不能推出空字则结束循环标识当前字符c已查找其FIRST集. 3. 计算FOLLOW集FOLLOW集的构造可用如下方法来求:对于文法中的符号X VN ,其FOLLOW(A)集合可反复应用下列规则计算,直到FOLLOW(A)集合不再增大为止。(1)对于文法开始符号S,因为S S,故#FOLLOW(S);(2)若A B,其中BVN,(VT VN)*、(VT VN)+,则FIRST()-FOLLOW(B);(3)若A B或A B ( ),则FOLLOW(A) FOLLOW(B)。 FOLLOW集的算法描述如下: void

5、 FOLLOW(int i) X为待求的非终结符把当前字符放到一临时数组foll中,标识求已求其FOLLOW集.避免循环递归if X为开始符号 then #FOLLOW(X) 对全部的产生式找一个右部含有当前字符X的产生式注:比如求FOLLOW(B)则找AX或AX()的产生式if X在产生式右部的最后(形如产生式AX) then查找非终结符A是否已经求过其FOLLOW集.避免循环递归if 非终结符A已求过其FOLLOW集 thenFOLLOW(A)FOLLOW(X)继续查下一条产生式是否含有Xelse求A之FOLLOW集,并标识为A已求其FOLLOW集else if X不在产生式右部的最后(形

6、如AB) thenif右部X后面的符号串能推出空字 then查找是否已经求过其FOLLOW集.避免循环递归if 已求过的FOLLOW集 then FOLLOW(A)FOLLOW(B)结束本次循环else if 不能推出空字 then 求FIRST()把FIRST()中所有非空元素加入到FOLLOW(B)中标识当前要求的非终结符X的FOLLOW集已求过4.计算SELECT集SELECT集的构造算法如下:对所有的规则产生式Ax:(1)若x不能推出空字,则SELECT(Ax) = FIRST(x);(2)若x可推出空字,则SELECT(Ax)=FIRST(x) FOLLOW(A)。算法描述如下:=产

7、生式总数-1;先把当前产生式右部的FIRST集(一切非空元素,不包括)放入到当前产生式的SELECT(i); if 产生式右部符号串可推出空字 then把i指向的当前产生式左部的非终结符号的FOLLOW集并入到SELECT(i)中5.判断是否LL(1)文法要判断是否为LL(1)文法,需要输入的文法G有如下要求:具有相同左部的规则的SELECT集两两不相交,即:SELECT(A) SELECT(A)= 如果输入的文法都符合以上的要求,则该文法可以用LL(1)方法分析。 把第一条产生式的SELECT(0)集放到一个临时数组temp中 for(i=1; 求temp的长度length if i指向的当

8、前产生式的左部等于上一条产生式的左部 then 把SELECT(i)并入到temp数组中 If temp的长度小于length加上SELECT (i)的长度 返回0 else把temp清空把SELECT (i)存放到temp中结果返回1;四、算法#includestdio.hstring.h/*/int count=0; /产生式的个数int number; /所有终结符和非终结符的总数char start; /开始符号char termin50; /终结符号char non_ter50; /非终结符号char v50; /所有符号char left50; /左部char right5050;

9、 /右部char first5050,follow5050; /各产生式右部的FIRST和左部的FOLLOW集合char first15050; /所有单个符号的FIRST集合char select5050; /各个产生式的SELECT集合char firstflag50,followflag50; /记录各符号的FIRST和FOLLOW是否已求过char empty20; /记录可推出&的符号char nonempty20; /记录不可推出&char empt20; /求_emp()时使用char TEMP50; /求FOLLOW时存放某一符号串的FIRST集合int validity=1;

10、 /表示输入文法是否有效int ll=1; /表示输入文法是否为LL(1)文法int M2020; /分析表char choose; /用户输入时使用char foll20; /求FOLLOW集合时使用/*判断一个字符c是否在指定字符串p中*/int in(char c,char *p) if(strlen(p)=0) return(0); if(pi=c) return(1); /若在,返回1 if(i=(int)strlen(p) return(0); /若不在,返回0将单个符号或符号串并入另一符号串void merge(char *d,char *s,int type) /是目标符号串,

11、s是源串,type1,源串中的一并并入目串; /type2,源串中的不并入目串 int i,j;=(int)strlen(s)-1; if(type=2&si=); for(j=0;j+) if(j(int)strlen(d)&si=dj) break; /若已存在,则退出,继续看下一个源串字符 if(j=(int)strlen(d) /若不存在,则并入 dj=si; dj+1= 读入一个文法char grammer(char *t,char *n,char *left,char right5050) char vn50,vt50; char s; char p5050; printf(请输入

12、文法的非终结符号串: scanf(%s,vn); getchar(); i=strlen(vn); memcpy(n,vn,i); ni=请输入文法的终结符号串:,vt); i=strlen(vt); memcpy(t,vt,i); ti=请输入文法的开始符号:%c,&s);请输入文法产生式的条数:%di); count=i; for(j=1;j) /检测输入错误 printf(n输入错误! validity=0; return( return(s);判断读入的文法是否正确int judge() if(in(lefti,non_ter)=0) /若左部不在非终结符中,报错n文法左部出错! fo

13、r(j=0;=(int)strlen(righti)-1; if(in(rightij,non_ter)=0&in(rightij,termin)=0&rightij!) /若右部某一符号不在非终结符、终结符中且不为,报错 printf(n文法右部出错! validity=0; return(0); return(1);求所有能直接推出&void emp(char c) if(righti0=c&strlen(righti)=1) temp0=lefti; temp1= merge(empty,temp,1);/求所有能直接推出的符号,结果保存到empty中 emp(lefti);求某一符号能

14、否推出 merge(empt,temp,1);/存放到一个临时数组empt里,标识此字符已查找其是否可推出空字 if(in(c,empty)=1)/如果c在可直接推出空字的empty中,返回1 return(1); if(lefti=c) /找一个左部为c的产生式 if(j=1&in(righti0,empty)=1)/右部长度为1且右部第一个字符在empty中.返回1(A- return(1); else if(j=1&in(righti0,termin)=1)/右部长度为1但第一个字符为终结符,返回0(A- continue; if(in(rightik,empt)=1)/查找临时数组em

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

16、空字,返回1求单个符号的FIRSTvoid 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= else if(in(c,non_ter)=1) /若为非终结符j+) /j为所有产生式中的序列 if(leftj=c) /找一个左部为c的产生式 if(in(rightj0,termin)=1|rightj0=

17、/若产生式右部第一个字符为终结符或空.-产生式Xa (aVT)或X&,则把a或&加进FIRST(X) temp0=rightj0; temp1= merge(first1i,temp,1); /-XY1Y2Yk的产生式,若Y1VN,则把FIRST(Y1)中的一切非空符号加进FIRST(X) else if(in(rightj0,non_ter)=1)/产生式右部第一个字符为非终结符 if(rightj0=c)/产生式右部的第一个符号等于当前字符,则跳到下一条产生式进行查找 continue; if(vk=rightj0)/求右部第一个字符在所有字符集中的位置k break; if(firstflagk=0 first2(k);/求其FIRST集 firstflagk=1/标识其为查找状态 merge(first1i,first1k,2);/求得结果并入到X的FIRST集.(int)strlen(rightj); empt0=/存放到一个临时数组里,标识此字符已查找其是否可推出空字 if(_emp(rightjk)=1&(int)strlen(rightj)-1) /当前产生式右部符号可推出空字,且当前字符不是右部的最后一个字符 for(m=0;m+) if(vm=rightjk+1)/获取右部符号下一个字符在所有字符集中的位置 break;

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

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