编译原理词法分析语法分析实验报告.docx

上传人:b****2 文档编号:890827 上传时间:2023-04-30 格式:DOCX 页数:55 大小:124.37KB
下载 相关 举报
编译原理词法分析语法分析实验报告.docx_第1页
第1页 / 共55页
编译原理词法分析语法分析实验报告.docx_第2页
第2页 / 共55页
编译原理词法分析语法分析实验报告.docx_第3页
第3页 / 共55页
编译原理词法分析语法分析实验报告.docx_第4页
第4页 / 共55页
编译原理词法分析语法分析实验报告.docx_第5页
第5页 / 共55页
编译原理词法分析语法分析实验报告.docx_第6页
第6页 / 共55页
编译原理词法分析语法分析实验报告.docx_第7页
第7页 / 共55页
编译原理词法分析语法分析实验报告.docx_第8页
第8页 / 共55页
编译原理词法分析语法分析实验报告.docx_第9页
第9页 / 共55页
编译原理词法分析语法分析实验报告.docx_第10页
第10页 / 共55页
编译原理词法分析语法分析实验报告.docx_第11页
第11页 / 共55页
编译原理词法分析语法分析实验报告.docx_第12页
第12页 / 共55页
编译原理词法分析语法分析实验报告.docx_第13页
第13页 / 共55页
编译原理词法分析语法分析实验报告.docx_第14页
第14页 / 共55页
编译原理词法分析语法分析实验报告.docx_第15页
第15页 / 共55页
编译原理词法分析语法分析实验报告.docx_第16页
第16页 / 共55页
编译原理词法分析语法分析实验报告.docx_第17页
第17页 / 共55页
编译原理词法分析语法分析实验报告.docx_第18页
第18页 / 共55页
编译原理词法分析语法分析实验报告.docx_第19页
第19页 / 共55页
编译原理词法分析语法分析实验报告.docx_第20页
第20页 / 共55页
亲,该文档总共55页,到这儿已超出免费预览范围,如果喜欢就下载吧!
下载资源
资源描述

编译原理词法分析语法分析实验报告.docx

《编译原理词法分析语法分析实验报告.docx》由会员分享,可在线阅读,更多相关《编译原理词法分析语法分析实验报告.docx(55页珍藏版)》请在冰点文库上搜索。

编译原理词法分析语法分析实验报告.docx

编译原理词法分析语法分析实验报告

编译原理实验报告

一.LL

(1)文法分析

1.设计要求

(1)对输入文法,它能判断是否为LL

(1)文法,若是,则转

(2);否则报错并终止;

(2)输入已知文法,由程序自动生成它的LL

(1)分析表;

(3)对于给定的输入串,应能判断识别该串是否为给定文法的句型。

2.分析

该程序可分为如下几步:

(1)读入文法

(2)判断正误

(3)若无误,判断是否为LL

(1)文法

(4)若是,构造分析表;

(5)由总控算法判断输入符号串是否为该文法的句型。

3.流程图

4.源程序

/*******************************************

语法分析程序

作者:

xxx

学号:

xxx

********************************************/

#include

#include

#include

/*******************************************/

intcount=0;/*分解的产生式的个数*/

intnumber;/*所有终结符和非终结符的总数*/

charstart;/*开始符号*/

chartermin[50];/*终结符号*/

charnon_ter[50];/*非终结符号*/

charv[50];/*所有符号*/

charleft[50];/*左部*/

charright[50][50];/*右部*/

charfirst[50][50],follow[50][50];/*各产生式右部的FIRST和左部的FOLLOW集合*/

charfirst1[50][50];/*所有单个符号的FIRST集合*/

charselect[50][50];/*各单个产生式的SELECT集合*/

charf[50],F[50];/*记录各符号的FIRST和FOLLOW是否已求过*/

charempty[20];/*记录可直接推出^的符号*/

charTEMP[50];/*求FOLLOW时存放某一符号串的FIRST集合*/

intvalidity=1;/*表示输入文法是否有效*/

intll=1;/*表示输入文法是否为LL

(1)文法*/

intM[20][20];/*分析表*/

charchoose;/*用户输入时使用*/

charempt[20];/*求_emp()时使用*/

charfo[20];/*求FOLLOW集合时使用*/

/*******************************************

判断一个字符是否在指定字符串中

********************************************/

intin(charc,char*p)

{

inti;

if(strlen(p)==0)

return(0);

for(i=0;;i++)

{

if(p[i]==c)

return

(1);/*若在,返回1*/

if(i==strlen(p))

return(0);/*若不在,返回0*/

}

}

/*******************************************

得到一个不是非终结符的符号

********************************************/

charc()

{

charc='A';

while(in(c,non_ter)==1)

c++;

return(c);

}

/*******************************************

分解含有左递归的产生式

********************************************/

voidrecur(char*point)

{/*完整的产生式在point[]中*/

intj,m=0,n=3,k;

chartemp[20],ch;

ch=c();/*得到一个非终结符*/

k=strlen(non_ter);

non_ter[k]=ch;

non_ter[k+1]='\0';

for(j=0;j<=strlen(point)-1;j++)

{

if(point[n]==point[0])

{/*如果‘|’后的首符号和左部相同*/

for(j=n+1;j<=strlen(point)-1;j++)

{

while(point[j]!

='|'&&point[j]!

='\0')

temp[m++]=point[j++];

left[count]=ch;

memcpy(right[count],temp,m);

right[count][m]=ch;

right[count][m+1]='\0';

m=0;

count++;

if(point[j]=='|')

{

n=j+1;

break;

}

}

}

else

{/*如果‘|’后的首符号和左部不同*/

left[count]=ch;

right[count][0]='^';

right[count][1]='\0';

count++;

for(j=n;j<=strlen(point)-1;j++)

{

if(point[j]!

='|')

temp[m++]=point[j];

else

{

left[count]=point[0];

memcpy(right[count],temp,m);

right[count][m]=ch;

right[count][m+1]='\0';

printf("count=%d",count);

m=0;

count++;

}

}

left[count]=point[0];

memcpy(right[count],temp,m);

right[count][m]=ch;

right[count][m+1]='\0';

count++;

m=0;

}

}

}

/*******************************************

分解不含有左递归的产生式

********************************************/

voidnon_re(char*point)

{

intm=0,j;

chartemp[20];

for(j=3;j<=strlen(point)-1;j++)

{

if(point[j]!

='|')

temp[m++]=point[j];

else

{

left[count]=point[0];

memcpy(right[count],temp,m);

right[count][m]='\0';

m=0;

count++;

}

}

left[count]=point[0];

memcpy(right[count],temp,m);

right[count][m]='\0';

count++;

m=0;

}

/*******************************************

读入一个文法

********************************************/

chargrammer(char*t,char*n,char*left,charright[50][50])

{

charvn[50],vt[50];

chars;

charp[50][50];

inti,j,k;

printf("\n请输入文法的非终结符号串:

");

scanf("%s",vn);

getchar();

i=strlen(vn);

memcpy(n,vn,i);

n[i]='\0';

printf("请输入文法的终结符号串:

");

scanf("%s",vt);

getchar();

i=strlen(vt);

memcpy(t,vt,i);

t[i]='\0';

printf("请输入文法的开始符号:

");

scanf("%c",&s);

getchar();

printf("请输入文法产生式的条数:

");

scanf("%d",&i);

getchar();

for(j=1;j<=i;j++)

{

printf("请输入文法的第%d条(共%d条)产生式:

",j,i);

scanf("%s",p[j-1]);

getchar();

}

for(j=0;j<=i-1;j++)

if(p[j][1]!

='-'||p[j][2]!

='>')

{printf("\ninputerror!

");

validity=0;

return('\0');

}/*检测输入错误*/

for(k=0;k<=i-1;k++)

{/*分解输入的各产生式*/

if(p[k][3]==p[k][0])

recur(p[k]);

else

non_re(p[k]);

}

return(s);

}

/*******************************************

将单个符号或符号串并入另一符号串

********************************************/

voidmerge(char*d,char*s,inttype)

{/*d是目标符号串,s是源串,type=1,源串中的‘^’一并并入目串;

type=2,源串中的‘^’不并入目串*/

inti,j;

for(i=0;i<=strlen(s)-1;i++)

{

if(type==2&&s[i]=='^')

;

else

{

for(j=0;;j++)

{

if(j

break;

if(j==strlen(d))

{

d[j]=s[i];

d[j+1]='\0';

break;

}

}

}

}

}

/*******************************************

求所有能直接推出^的符号

********************************************/

voidemp(charc)

{/*即求所有由‘^’推出的符号*/

chartemp[10];

inti;

for(i=0;i<=count-1;i++)

{

if(right[i][0]==c&&strlen(right[i])==1)

{

temp[0]=left[i];

temp[1]='\0';

merge(empty,temp,1);

emp(left[i]);

}

}

}

/*******************************************

求某一符号能否推出‘^’

********************************************/

int_emp(charc)

{/*若能推出,返回1;否则,返回0*/

inti,j,k,result=1,mark=0;

chartemp[20];

temp[0]=c;

temp[1]='\0';

merge(empt,temp,1);

if(in(c,empty)==1)

return

(1);

for(i=0;;i++)

{

if(i==count)

return(0);

if(left[i]==c)/*找一个左部为c的产生式*/

{

j=strlen(right[i]);/*j为右部的长度*/

if(j==1&&in(right[i][0],empty)==1)

return

(1);

elseif(j==1&&in(right[i][0],termin)==1)

return(0);

else

{

for(k=0;k<=j-1;k++)

if(in(right[i][k],empt)==1)

mark=1;

if(mark==1)

continue;

else

{

for(k=0;k<=j-1;k++)

{

result*=_emp(right[i][k]);

temp[0]=right[i][k];

temp[1]='\0';

merge(empt,temp,1);

}

}

}

if(result==0&&i

continue;

elseif(result==1&&i

return

(1);

}

}

}

/*******************************************

判断读入的文法是否正确

********************************************/

intjudge()

{

inti,j;

for(i=0;i<=count-1;i++)

{

if(in(left[i],non_ter)==0)

{/*若左部不在非终结符中,报错*/

printf("\nerror1!

");

validity=0;

return(0);

}

for(j=0;j<=strlen(right[i])-1;j++)

{

if(in(right[i][j],non_ter)==0&&in(right[i][j],termin)==0&&right[i][j]!

='^')

{/*若右部某一符号不在非终结符、终结符中且不为‘^’,报错*/

printf("\nerror2!

");

validity=0;

return(0);

}

}

}

return

(1);

}

/*******************************************

求单个符号的FIRST

********************************************/

voidfirst2(inti)

{/*i为符号在所有输入符号中的序号*/

charc,temp[20];

intj,k,m;

c=v[i];

charch='^';

emp(ch);

if(in(c,termin)==1)/*若为终结符*/

{

first1[i][0]=c;

first1[i][1]='\0';

}

elseif(in(c,non_ter)==1)/*若为非终结符*/

{

for(j=0;j<=count-1;j++)

{

if(left[j]==c)

{

if(in(right[j][0],termin)==1||right[j][0]=='^')

{

temp[0]=right[j][0];

temp[1]='\0';

merge(first1[i],temp,1);

}

elseif(in(right[j][0],non_ter)==1)

{

if(right[j][0]==c)

continue;

for(k=0;;k++)

if(v[k]==right[j][0])

break;

if(f[k]=='0')

{

first2(k);

f[k]='1';

}

merge(first1[i],first1[k],2);

for(k=0;k<=strlen(right[j])-1;k++)

{

empt[0]='\0';

if(_emp(right[j][k])==1&&k

{

for(m=0;;m++)

if(v[m]==right[j][k+1])

break;

if(f[m]=='0')

{

first2(m);

f[m]='1';

}

merge(first1[i],first1[m],2);

}

elseif(_emp(right[j][k])==1&&k==strlen(right[j])-1)

{

temp[0]='^';

temp[1]='\0';

merge(first1[i],temp,1);

}

else

break;

}

}

}

}

}

f[i]='1';

}

/*******************************************

求各产生式右部的FIRST

********************************************/

voidFIRST(inti,char*p)

{

intlength;

intj,k,m;

chartemp[20];

length=strlen(p);

if(length==1)/*如果右部为单个符号*/

{

if(p[0]=='^')

{

if(i>=0)

{

first[i][0]='^';

first[i][1]='\0';

}

else

{

TEMP[0]='^';

TEMP[1]='\0';

}

}

else

{

for(j=0;;j++)

if(v[j]==p[0])

break;

if(i>=0)

{

memcpy(first[i],first1[j],strlen(first1[j]));

first[i][strlen(first1[j])]='\0';

}

else

{

memcpy(TEMP,first1[j],strlen(first1[j]));

TEMP[strlen(first1[j])]='\0';

}

}

}

else/*如果右部为符号串*/

{

for(j=0;;j++)

if(v[j]==p[0])

break;

if(i>=0)

merge(first[i],first1[j],2);

else

merge(TEMP,first1[j],2);

for(k=0;k<=length-1;k++)

{

empt[0]='\0';

if(_emp(p[k])==1&&k

{

for(m=0;;m++)

if(v[m]==right[i][k+1])

break;

if(i>=0)

merge(first[i],first1[m],2);

else

merge(TEMP,first1[m],2);

}

elseif(_emp(p[k])==1&&k==length-1)

{

temp[0]='^';

temp[1]='\0';

if(i>=0)

merge(first[i],temp,1);

else

merge(TEMP,temp,1);

}

elseif(_emp(p[k])==0)

break;

}

}

}

/*******************************************

求各产生式左部的FOLLOW

********************************************/

voidFOLLOW(inti)

{

intj,k,m,n,result=1;

charc,temp[20];

c=non_ter[i];/*c为待求的非终结符*/

temp[0]=c;

temp[1]='\0';

merge(fo,temp,1);

if(c==start)

{/*若为开始符号*/

temp[0]='#';

temp[1]='\0';

merge(follow[i],temp,1);

}

for(j=0;j<=count-1;j++)

{

if(in(c,right[j])==1)/*找一个右部含有c的产生式*/

{

for(k=0;;k++)

if(right[j][k]==c)

break;/*k为c在该产生式右部的序号*/

for(m=0;;m++)

if(v[m]==left[j])

break;/*m为产生式左部非终结符在所有符号中的序号*/

if(k==strlen(right[j])-1)

{/*如果c在产生式右部的最后*/

if(in(v[m],fo)==1)

{

merge(follow[i],follow[m],1);

continue;

}

if(F[m]=='0')

{

FOLLOW(m);

F[m]='1';

}

merge(follow[i],follow[m],1);

}

els

展开阅读全文
相关资源
猜你喜欢
相关搜索
资源标签

当前位置:首页 > 临时分类 > 批量上传

copyright@ 2008-2023 冰点文库 网站版权所有

经营许可证编号:鄂ICP备19020893号-2