1、编译原理语法分析器实验报告模板课程设计信安课 程 设 计 报 告课程名称 编译原理 实验项目 设计与实现一个词法分析器 实验仪器 PC机 北京信息科技大学信息管理学院(课程上机)实验报告实验名称设计与实现一个语法分析器实验地点信管机房204实验时间2011.12.201. 实验目的:结合讲授内容,进一步培养学生编译器设计的思想,加深对编译原理和应用程序的理解,针对编译过程的重点和难点内容进行编程,独立完成有一定工作量的语法分析程序设计任务,同时强调好的程序设计风格。2. 实验内容:对输入的文法判定是否为LL(1)文法,若是LL(1)文法,则构造LL(1)分析表,输入一个句子,依据LL(1)分析
2、表输出与句子对应的语法树。设计LL(1)文法的词法分析器。3. 实验要求:(1) 输入一个文法G(如:测试数据1);(2) 编写求FIRST集的算法;输出文法G的First集;(3) 编写求FOLLOW集的算法;输出文法G的FOLLOW集;(4) 编写求SELECT的算法,输出文法G的SELECT集;(5) 判定是否为LL(1)文法,若是,则编写构造LL(1)分析表的算法,并输出预测分析表(6) 编写表驱动的预测分析算法;(7) 给出输入一个句子(如:测试数据2)的预测分析步骤; (8) 输出依据句子构对应的语法树的过程; (9) 构造文法G的递归子程序;(选作)测试数据1:(若用其它测试数据
3、,请替换下面内容)输入文法G:S-aHH-aMd|dM-Ab|A-aM|e测试数据2:输入句子: aaabd#4. 实验准备:5. 实验过程:#include#include#include/*/int count=0; /*分解的产生式的个数*/int number; /*所有终结符和非终结符的总数*/char start; /*开始符号*/char termin50; /*终结符号*/char non_ter50; /*非终结符号*/char v50; /*所有符号*/char left50; /*左部*/char right5050; /*右部*/char first5050,follo
4、w5050; /*各产生式右部的FIRST和左部的FOLLOW集合*/char first15050; /*所有单个符号的FIRST集合*/char select5050; /*各单个产生式的SELECT集合*/char f50,F50; /*记录各符号的FIRST和FOLLOW是否已求过*/char empty20; /*记录可直接推出的符号*/char TEMP50; /*求FOLLOW时存放某一符号串的FIRST集合*/int validity=1; /*表示输入文法是否有效*/int ll=1; /*表示输入文法是否为LL(1)文法*/int M2020; /*分析表*/char ch
5、oose; /*用户输入时使用*/char empt20; /*求_emp()时使用*/char fo20; /*求FOLLOW集合时使用*/ /* 判断一个字符是否在指定字符串中*/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=strlen(p) return(0); /*若不在,返回0*/ /*得到一个不是非终结符的符号*/char c() char c=A; while(in(c,non_ter)=1) c+; return(
6、c);/*分解含有左递归的产生式*/void recur(char *point) /*完整的产生式在point中*/ int j,m=0,n=3,k; char temp20,ch; ch=c(); /*得到一个非终结符*/ k=strlen(non_ter); non_terk=ch; non_terk+1=0; for(j=0;j=strlen(point)-1;j+) if(pointn=point0) /*如果|后的首符号和左部相同*/ for(j=n+1;j=strlen(point)-1;j+) while(pointj!=|&pointj!=0) tempm+=pointj+;
7、 leftcount=ch; memcpy(rightcount,temp,m); rightcountm=ch; rightcountm+1=0; m=0; count+; if(pointj=|) n=j+1; break; else /*如果|后的首符号和左部不同*/ leftcount=ch; rightcount0=; rightcount1=0; count+; for(j=n;j=strlen(point)-1;j+) if(pointj!=|) tempm+=pointj; else leftcount=point0; memcpy(rightcount,temp,m); ri
8、ghtcountm=ch; rightcountm+1=0; printf( count=%d ,count); m=0; count+; leftcount=point0; memcpy(rightcount,temp,m); rightcountm=ch; rightcountm+1=0; count+; m=0; void non_re(char *point) int m=0,j; char temp20; for(j=3;j=strlen(point)-1;j+) if(pointj!=|) tempm+=pointj; else leftcount=point0; memcpy(r
9、ightcount,temp,m); rightcountm=0; m=0; count+; leftcount=point0; memcpy(rightcount,temp,m); rightcountm=0; count+; m=0;/*读入一个文法*/char grammer(char *t,char *n,char *left,char right5050) char vn50,vt50; char s; char p5050; int i,j,k; printf(请输入文法的非终结符号串:); scanf(%s,vn); getchar(); i=strlen(vn); memcpy
10、(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(); for(j=0;ji;j+) printf(请输入文法的第%d条(共%d条)产生式:,j,i); scanf(%s,pj);/二维数组首地址传递 平p,pi第i行第0列首地址,p+i表示第i行首地址 if( strc
11、mp(pj, xxx )=0) / j=j-2; getchar(); for(j=0;j) printf(ninput error!); validity=0; return(0); /*检测输入错误*/ for(k=0;k=i-1;k+) /*分解输入的各产生式*/ if(pk3=pk0) recur(pk); else non_re(pk); return(s);/*将单个符号或符号串并入另一符号串*/void merge(char *d,char *s,int type) /*d是目标符号串,s是源串,type1,源串中的 一并并入目串; type2,源串中的 不并入目串*/ int
12、i,j; for(i=0;i=strlen(s)-1;i+) if(type=2&si=) ; else for(j=0;j+) if(jstrlen(d)&si=dj) break; if(j=strlen(d) dj=si; dj+1=0; break; /*求所有能直接推出的符号*/void emp(char c) /*即求所有由 推出的符号*/ char temp10; int i; for(i=0;iB B- return(1); else if(j=1&in(righti0,termin)=1&lefti+1!=c)/A-a ,A-sB ,S- ,B- return(0); els
13、e if(j=1&in(righti0,termin)=1&lefti+1=c) continue; else for(k=0;kA mark=1; if(mark=1) continue; else for(k=0;k=j-1;k+) result*=_emp(rightik);/ 太漂亮了 temp0=rightik; temp1=0; merge(empt,temp,1); if(result=0&iBa ,A-S ,S- ,B!- 找下一个做不为的-C continue; else if(result=1&icount) return(1); /*判断读入的文法是否正确*/int ju
14、dge() int i,j; for(i=0;i=count-1;i+) if(in(lefti,non_ter)=0) /*若左部不在非终结符中,报错*/ printf(nerror1!); validity=0; return(0); for(j=0;j=strlen(righti)-1;j+) if(in(rightij,non_ter)=0&in(rightij,termin)=0&rightij!=) /*若右部某一符号不在非终结符、终结符中且不为 ,报错*/ printf(nerror2!); validity=0; return(0); return(1);/*求单个符号的FIR
15、ST 求出每个符号的first 并且将其存入first1*/void first2(int i) /*i为符号在所有输入符号中的序号*/ char c,temp20; int j,k,m; c=vi; char ch=; emp(ch); if(in(c,termin)=1) /*若为终结符*/ first1i0=c; first1i1=0; else if(in(c,non_ter)=1) /*若为非终结符*/ for(j=0;j或者A-a temp0=rightj0; temp1=0; merge(first1i,temp,1); else if(in(rightj0,non_ter)=1
16、) /左部非终结符并且右部为非终结符 if(rightj0=c)/A-A.寻找下一个A continue; for(k=0;k+) /A-B。无限循环,找到B的位置 if(vk=rightj0) break; if(fk=0) first2(k);/M-A. 此时找出A的位置在M 之后,而A 的first没有求过,所以要返回求 fk=1; merge(first1i,first1k,2);/A-B 则将B对应符号表位置找出,然后将B first1 复制给A for(k=0;k=strlen(rightj)-1;k+)/二位数组 strlen(rightj)表示第j行的长度 empt0=0; i
17、f(_emp(rightjk)=1&kAB A- 则找到B的位置,求B的first break; if(fm=0) first2(m); fm=1; merge(first1i,first1m,2); else if(_emp(rightjk)=1&k=strlen(rightj)-1) temp0=; temp1=0; merge(first1i,temp,1); else break; fi=1;/*求各产生式右部的FIRST*/void FIRST(int i,char *p) int length; int j,k,m; char temp20; length=strlen(p); i
18、f(length=1) /*如果右部为单个符号*/ if(p0=) if(i=0) firsti0=; firsti1=0; else TEMP0=; TEMP1=0; else for(j=0;j+) if(vj=p0) break; if(i=0) memcpy(firsti,first1j,strlen(first1j); firstistrlen(first1j)=0; else memcpy(TEMP,first1j,strlen(first1j); TEMPstrlen(first1j)=0; else /*如果右部为符号串*/ for(j=0;j+) if(vj=p0) break; if(i=0) merge(firsti,first1j,2); else merge(TEMP,first1j,2); for(k=0;k=length-1;k+) empt0=0; if(_emp(pk)=1&k=0) merge(firsti,first1m,2); else merge(TEMP,first1m,2); else
copyright@ 2008-2022 冰豆网网站版权所有
经营许可证编号:鄂ICP备2022015515号-1