语法分析器实验报告Word文档格式.docx
《语法分析器实验报告Word文档格式.docx》由会员分享,可在线阅读,更多相关《语法分析器实验报告Word文档格式.docx(26页珍藏版)》请在冰豆网上搜索。
(4)
(5)
该程序可分为如下几步:
读入文法
判断正误
若无误,判断是否为LL
(1)文法若是,构造分析表;
由句型判别算法判断输入符号串是为该文法的句型。
三、核心思想
该分析程序有15部分组成:
(1)首先定义各种需要用到的常量和变量;
(2)判断一个字符是否在指定字符串中;
读入一个文法;
将单个符号或符号串并入另一符号串;
求所有能直接推出&
的符号;
求某一符号能否推出‘&
'
;
判断读入的文法是否正确;
求单个符号的FIRST;
求各产生式右部的FIRST;
求各产生式左部的FOLLOW;
判断读入文法是否为一个LL
(1)文法;
构造分析表M;
句型判别算法;
一个用户调用函数;
主函数;
(6)
(7)
(8)
(9)
(10)
(11)
(12)
(13)
(14)
(15)下面是其中几部分程序段的算法思想:
1、求能推出空的非终结符集
I、实例中求直接推出空的empty集的算法描述如下:
voidemp(charc){参数c为空符号chartemp[10];
定义临时数组inti;
1,temp中
&
符号的数组empty中。
for(i=0;
i<
=count-1;
i++)从文法的第一个产生式开始查找{
if产生式右部第一个符号是空符号并且右部长度为then将该条产生式左部符号保存在临时数组将临时数组中的元素合并到记录可推出
}
n、求某一符号能否推出&
int_emp(charc)
{//若能推出&
,返回1;
否则,返回0
inti,j,k,result=1,mark=0;
chartemp[20];
temp[0]=c;
temp[1]='
\0'
;
存放到一个临时数组empt里,标识此字符已查找其是否可推出空字如果c在可直接推出空字的empty[]中,返回1for(i=0;
i++)
{if(i==count)
return(0);
//j为c所在产生式右部的长度
1且右部第一个字符在empty[]中.then返回1(A->
B,B可推出空)
1但第一个字符为终结符,then返回0(A->
a,a为终结符)
找一个左部为c的产生式j=strlen(right[i]);
if右部长度为if右部长度为
else
{
for(k=0;
k<
=j-1;
k++)
查找临时数组empt[].并标记mark-=1(A->
AB)
if找到的字符与当前字符相同(A->
AB)结束本次循环
else(mark等于0)
查找右部符号是否可推出空字,把返回值赋给result
if
把当前符号加入到临时数组empt[]里.
当前字符不能推出空字且还没搜索完全部的产生式then跳出本次循环继续搜索下一条产生式elseif//当前字符可推出空字,返回
2、计算每个符号的first集:
实例中求单个符号的FIRST集的算法描述如下:
voidfirst2(inti){参数i为符号在所有输入符号中的序号c等于指示器i所指向的符号在保存终结符元素的termin[]数组查找c
ifc为终结符(c€Vt),thenFIRST(c)={c}在保存终结符元素的non_ter[]数组查找cifc是非终结符(c€Vn)
(即C7a(a€Vt)或&
)then
then
在所有产生式中查找c所在的产生式if产生式右部第一个字符为终结符或空把a或&
加进FIRST(c)if产生式右部第一个字符为非终结符
if产生式右部的第一个符号等于当前字符then跳到下一条产生式进行查找求当前非终结符在所有字符集中的位置if当前非终结符还没求其FIRST集then查找它的FIRST集并标识此符号已求其FIRST集求得结果并入到c的FIRST集.
if当前产生式右部符号可推出空字且当前字符不是右部的最后一个字符获取右部符号下一个字符在所有字符集中的位置if此字符的FIRST集还未查找then找其FIRST集,并标其查找状态为1把求得的FIRST集并入到c的FIRST集.
if当前右部符号串可推出空且是右部符号串的最后一个字符(即产生式为c7
丫1丫2…Yk,若对一切1<
=i<
=k,均有&
€FIRST(Yi),则将&
€符号加进FIRST(c))then
把空字加入到当前字符c的FIRST集.
不能推出空字则结束循环
标识当前字符c已查找其FIRST集.}
3.计算FOLLOW
FOLLOW集的构造可用如下方法来求:
对于文法中的符号XVn,其FOLLOW(A)集合可反复应用下列规则计算,直到
FOLLOW(A)集合不再增大为止。
(1)对于文法开始符号S,因为SS,故#FOLLOW(S);
(2)若A7B,其中BVn,(VtVn)*、(VtVn)+,贝U
FIRST()-{}FOLLOW(B);
(3)若A7B或A7B(禺),贝y
FOLLOW(A)FOLLOW(B)。
FOLLOW集的算法描述如下:
voidFOLLOW(inti)
X为待求的非终结符
把当前字符放到一临时数组fo叩中,标识求已求其FOLLOW集.避免循环递归
ifX为开始符号then#€FOLLOW(X)
对全部的产生式找一个右部含有当前字符X的产生式
注:
比如求FOLLOW(B)则找A7aX或A7X(兰>
£
)的产生式
ifX在产生式右部的最后(形如产生式A7X)then查找非终结符A是否已经求过其FOLLOW集.避免循环递归
if非终结符A已求过其FOLLOW集then
FOLLOW(A)€FOLLOW(X)
继续查下一条产生式是否含有X
求A之FOLLOW集,并标识为A已求其FOLLOW集
elseifX不在产生式右部的最后(形如A7B)then
if右部X后面的符号串能推出空字then
查找是否已经求过其FOLLOW集.避免循环递归
if已求过的FOLLOW集then
FOLLOW(A)€FOLLOW(B)
结束本次循环
elseif不能推出空字then
求FIRST()
把FIRST()中所有非空元素加入到FOLLOW(B)中
标识当前要求的非终结符X的FOLLOW集已求过
4.计算SELECT!
SELECT集的构造算法如下:
对所有的规则产生式A7X:
(1)若X不能推出空字,贝ySELECT(A7X)=FIRST(x);
(2)若X可推出空字,贝USELECT(A7x)=FIRST(x)-{}FOLLOW(A)。
算法描述如下:
=产生式总数-1;
先把当前产生式右部的FIRST集(一切非空元素,不包括£
放入到当前产生式的
SELECT(i);
if产生式右部符号串可推出空字then
把i指向的当前产生式左部的非终结符号的FOLLOW集并入到SELECT(i)中
5.判断是否LL
(1)文法
要判断是否为LL
(1)文法,需要输入的文法G有如下要求:
具有相同左部的规则的SELECT集两两不相交,即:
SELECT(AT)nSELECT(A)=如果输入的文法都符合以上的要求,则该文法可以用算法描述如下:
把第一条产生式的SELECT(0)集放到一个临时数组temp[]中for(i=1;
=产生式总数-1;
求temp的长度length
ifi指向的当前产生式的左部等于上一条产生式的左部temp数组中length加上SELECT(i)的长度
LL
(1)方法分析。
把SELECT(i)并入到
Iftemp的长度小于返回0
把temp清空
把SELECT(i)存放到
结果返回1;
四、算法
temp中
#include<
stdlib.h>
stdio.h>
string.h>
/*******************************************/intcount=0;
intnumber;
charstart;
chartermin[50];
charnon_ter[50];
charv[50];
charleft[50];
charright[50][50];
//产生式的个数
//所有终结符和非终结符的总数
//开始符号
//终结符号
//非终结符号
//所有符号
//左部
//右部
charfirst[50][50],follow[50][50];
charfirst1[50][50];
charselect[50][50];
charfirstflag[50],followflag[50];
charempty[20];
charnonempty[20];
charempt[20];
charTEMP[50];
intint
//各产生式右部的FIRST和左部的FOLLOW集合//所有单个符号的FIRST集合//各个产生式的SELECT集合
//记录各符号的FIRST和FOLLOW是否已求过//记录可推出&
的符号
//记录不可推出&
II求_emp()时使用
//求FOLLOW时存放某一符号串的FIRST集合
int
validity=1;
ll=1;
M[20][20];
II表示输入文法是否有效
II表示输入文法是否为LL
(1)文法
II分析表
charchoose;
//用户输入时使用
charfoll[20];
//求FOLLOW集合时使用
/*******************************************
判断一个字符c是否在指定字符串p中********************************************/
intin(charc,char*p){