编译原理课程设计报告简单编译器实现.docx
《编译原理课程设计报告简单编译器实现.docx》由会员分享,可在线阅读,更多相关《编译原理课程设计报告简单编译器实现.docx(37页珍藏版)》请在冰点文库上搜索。
编译原理课程设计报告简单编译器实现
编译原理-课程设计报告-简单编译器实现
成绩:
课程设计
题目:
简单编译器实现
学院:
信息工程学院计算机系
专业:
计算机科学与技术
班级:
计科1103班
组长:
小组成员:
指导教师:
1概述
编译程序的工作过程一般可以分为五个阶段:
词法分析、语法分析、语义分析与中间代码产生、优化、目标代码生成。
每一个阶段在功能上是相对独立的,它一方面从上一个阶段获取分析的结果来进行分析,另一方面由将结果传递给下一个阶段。
由编译程序的五个阶段就对应了编译系统的结构。
其中词法分析器利用超前搜索、状态转换等方法,将源程序转化成为一个一个的单词符号二元式。
一般程序语言的单词符号包括关键字、运算符、常数、标识符和界符。
语法分析器将这些单词符号作为输入,对它进行语法分析。
语法分析分为两种方法:
自上而下分析法和自下而上分析法。
针对不同程序语言的语法规则可以采取不同的分析方法,当然两种方法也可以同时使用。
语法分析器把语法单元作为输入供语义分析器使用。
一般的语义分析器主要采用的是语法制导方法,即在语法分析的同时进行语法分析,并产生一定的语义动作,来生成中间代码。
上面三个过程可以与硬件无关,而接下来的优化器和目标代码生成器是针对某一种处理器而言的。
代码优化是将语义分析生成的中间代码进行优化,产生执行效率更高的代码。
目标代码生成器最终生成可以在某种机器上运行的机器语言或者汇编语言。
在整个编译过程中还包括对表格的操作和对错误的处理,这些也都是非常重要的环节。
1.1源、目标语言简介
使用C语言做简单语法分析器,C语言是一门高级计算机编程语言,设计目标是提供一种能以简易的方式编译、处理低级存储器、产生少量的机器码以及不需要任何运行环境支持便能运行的编程语言
1.2实现平台与运行平台简介
在win32环境下进行编译,Win32是指MicrosoftWindows操作系统的32位环境,是目前使用最多的操作系统。
实验环境:
需要TC、VC++6.0等开发工具作为本次试验的环境。
1.3其它
通过实现一个可以把类似c语言的源代码转变为中间代码的编译器,更好地理解编译的过程,锻炼我们组的编程能力。
2简单词法分析器的设计与实现
2.1基础理论说明
词法分析负责对源程序的字符串进行扫描和分解,根据构词法将字符流(CharacterStream)转化成单词流(TokenStream)。
2.2需求分析
词法分析器产生下述小语言的单词序列
这个小语言的所有的单词符号,以及它们的种别编码和内部值[1]如下表:
单词符号
种别编码
助记符
内码值
DIM
IF
DO
STOP
END
标识符
常数(整)
=
+
*
**
,
(
)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
$DIM
$IF
$DO
$STOP
$END
$ID
$INT
$ASSIGN
$PLUS
$STAR
$POWER
$COMMA
$LPAR
$RPAR
-
-
-
-
-
-
内部字符串
标准二进形式
-
-
-
-
-
-
2.3概要设计
首先,所有的关键字(如IF﹑WHILE等)都是“保留字”。
所谓的保留字的意思是,用户不得使用它们作为自己定义的标示符。
例如,下面的写法是绝对禁止的:
IF(5)=x
其次,由于把关键字作为保留字,故可以把关键字作为一类特殊标示符来处理。
也就是说,对于关键字不专设对应的转换图。
但把它们(及其种别编码)预先安排在一张表格中(此表叫作保留字表)。
当转换图识别出一个标识符时,就去查对这张表,确定它是否为一个关键字。
再次,如果关键字、标识符和常数之间没有确定的运算符或界符作间隔,则必须至少用一个空白符作间隔(此时,空白符不再是完全没有意义的了)。
例如,一个条件语句应写为
IFi>0i=1;
而绝对不要写成
IFi>0i=1;
因为对于后者,我们的分析器将无条件地将IFI看成一个标识符。
2.4详细设计
状态转换图[2]
词法分析器的流程图[3]
2.5测试数据与结果
2.6心得体会
设计该词法分析器的过程中虽然没有实际将所有的状态转移表建立出来,但是所用的思想是根据状态转移表实现对单词的识别。
首先构造一个保留字表,然后,每输入一个字符就检测应该进入什么状态,并将该字符连接到d串后继续输入,如此循环,最后根据所在的接受状态以及保留字表识别单词
3简单语法分析器设计与实现
3.1基础理论说明
在计算机科学和语言学中,语法分析(英:
Syntacticanalysis,也叫Parsing)是根据某种给定的形式文法对由单词序列(如英语单词序列)构成的输入文本进行分析并确定其语法结构的一种过程。
语法分析器(Parser)通常是作为编译器或解释器的组件出现的,它的作用是进行语法检查、并构建由输入的单词组成的数据结构(一般是语法分析树、抽象语法树等层次化的数据结构)。
语法分析器通常使用一个独立的词法分析器从输入字符流中分离出一个个的“单词”,并将单词流作为其输入。
实际开发中,语法分析器可以手工编写,也可以使用工具(半)自动生成。
3.2需求分析
语法分析是编译过程的核心部分。
它的任务是在词法分析识别出单词符号串的基础上,分析并判定程序的语法结构是否符合语法规则。
语法分析器的工作本质上是按文法的产生式,识别输入串是否是一个句子。
自上而下分析法的主旨是,对任何输入串,试图用一切可能的方法,从文法开始符号出发,自上而下地为输入串建立一棵语法树。
这种方法本质上是一种试探过程,是反复使用不同产生式谋求匹配输入串的过程。
3.3概要设计
语法分析器能识别由加+减-乘*除/乘方^括号()操作数所组成的算术表达式,其文法如下:
E→E+T|E-T|T
T→T*F|T/F|F
F→P^F|P
p→(E)|i
使用的算法可以是:
预测分析法;递归下降分析法;算符优先分析法;LR分析法等
3.4详细设计
语法分析器主程序图[4]
3.5测试数据与结果
3.6心得体会
此次实验,让我们组对编译原理的基本知识有了深入的了解,加强了对语法分析的认识。
代码的编写过程中用到了一些以前从未用过的函数,都是现学现用,掌握还不是很深。
在代码调试过程中结果出现许多无法解释的错误,但仍旧坚持下来了,最终调试出了结果。
通过这次实验,我们组的动手实践能力得到很大的提高。
4中间代码产生器的设计与实现
4.1基础理论说明
在进行了语法分析和语义分析阶段的工作之后,有的编译程序将源程序变成一种内部表示形式,这种内部表示形式叫做中间语言或中间表示或中间代码。
所谓“中间代码”是一种结构简单、含义明确的记号系统,这种记号系统复杂性介于源程序语言和机器语言之间,容易将它翻译成目标代码。
另外,还可以在中间代码一级进行与机器无关的优化。
产生中间代码的过程叫中间代码生成。
4.2需求分析
定义一种语言除了要求定义语法外,还要求定义语义,即对语言的各种语法单位赋予具体的意义。
语义分析的任务是首先对每种语法单位进行静态的语义审查,然后分析其含义,并用另一种语言形式,即比源语言更加接近于目标语言的一种中间代码来进行描述这种语言。
因此,中间代码就显得十分重要,它关系着整个程序语言的正确编译与否,同时也是进行下一步编译的重要先决条件。
4.3概要设计
产生上述算术表达式的中间代码(四元式序列)
递归下降子程序:
数据结构:
SYN—算符栈;
SEM—语义栈;
4.4详细设计
中间代码生成器流程图[5]:
4.5测试数据与结果
4.6心得体会
我们知道,定义一种语言除了要求定义语法外,还要求定义语义,即对语言的各种语法单位赋予具体的意义。
语义分析的任务是首先对每种语法单位进行静态的语义审查,然后分析其含义,并用另一种语言形式,即比源语言更加接近于目标语言的一种中间代码来进行描述这种语言。
因此,中间代码就显得十分重要,它关系着整个程序语言的正确编译与否,同时也是进行下一步编译的重要先决条件。
参考文献
[1]AlfredD.Ullman.Compilers:
Principles,Techniques,andTools:
4页
[2]zjbujs.XX文库.编译原理词法语法语义分析器设计.2013-07-23:
5页
[3]zjbujs.XX文库.编译原理词法语法语义分析器设计.2013-07-23:
6页
[4]线性大树.XX文库.编译原理词法语法语义设计实现.2014-06-06:
8页
[5]LWH1989216.XX文库.编译原理词法分析和语法分析(C语言版)2011-05-11:
10页
附录:
附录A:
主要源程序与系统截图
/*******************************词法分析器源代码******************************/
#include
#include
#include
#defineMAX150//词法分析表的最大容量
#defineMAXBUF255//缓冲区的最大缓冲量
charprog[MAXBUF],token[MAX];
charch;
intsyn,p,m,n,sum;
char*rwtab[6]={"begin","if","then","while","do","end"};
///////////////////////////////////////////////
//词法分析程序
///////////////////////////////////////////////
voidscaner()
{
for(m=0;mtoken[m]=NULL;
m=0;sum=0;
ch=prog[p++];
while(ch=='')
ch=prog[p++];//读取下一个字符;
if(ch>=65&&ch<=122/*是字母字符*/)
{
while(ch>=65&&ch<=122||ch>=48&&ch<=57)/*为字母字符或数字字符*/
{
token[m++]=ch;
ch=prog[p++];//读取下一个字符;
}
token[m++]='\0';
p=p-1;
syn=10;
for(n=0;n<6;n++)
if(strcmp(token,rwtab[n])==0)
{
syn=n+1;//给出syn值;
break;
}
}
elseif(ch>=48&&ch<=57/*ch为数字字符*/)
{
while(ch>=48&&ch<=57/*ch为数字字符*/)
{
sum=sum*10+ch-'0';
ch=prog[p++];//读取下一个字符;
}
p=p-1;//回退一个字符;
syn=11;
}
elseswitch(ch)
{
case'<':
m=0;token[m++]=ch;
ch=prog[p++];//读取下一个字符;
if(ch=='>')
{
syn=21;
token[m++]=ch;
}
elseif(ch=='=')
{
syn=22;
token[m++]=ch;
}
else
{
syn=20;
p=p-1;//回退一个字符;
}
break;
case'>':
token[m++]=ch;;
ch=prog[p++];//读取下一个字符;
if(ch=='=')
{
syn=24;//将>=的中别码=>syn;
token[m++]=ch;;
}
else
{
syn=23;
p=p-1;//回退一个字符;
}
break;
case':
':
token[m++]=ch;;
ch=prog[p++];//读取下一个字符;
if(ch=='=')
{
syn=18;
token[m++]=ch;;
}
else
{
syn=17;
p=p-1;//回退一个字符;
}
break;
case'+':
syn=13;token[0]=ch;
break;
case'-':
syn=14;token[0]=ch;
break;
case'*':
syn=15;token[0]=ch;
break;
case'/':
syn=16;token[0]=ch;
break;
case'=':
syn=25;token[0]=ch;
break;
case';':
syn=26;token[0]=ch;
break;
case'(':
syn=27;token[0]=ch;
break;
case')':
syn=28;token[0]=ch;
break;
case'#':
syn=0;token[0]=ch;
break;
default:
syn=-1;
break;
}
}
///////////////////////////////////////////
//主函数
///////////////////////////////////////////
voidmain()
{
charA;
cout<<"*****************************************"<loop:
p=0;
cout<<"*****************************************"<printf("pleaseinputstring(以#结束):
\n");
do
{
scanf("%c",&ch);
prog[p++]=ch;//输入源程序字符串,送到缓冲区prog[p++]中;
}
while(ch!
='#');
p=0;
do
{
scaner();
switch(syn)
{
case11:
cout<<"("<break;
case-1:
cout<<"error"<break;
default:
cout<<"("<}
}while(syn!
=0);
cout<<"*****************************************"<cout<<"请确定是否继续使用程序:
S为继续;其它为退出;"<cout<<"是否继续:
";
cin>>A;
switch(A)
{
case'S':
gotoloop;
default:
cout<<"*****************************************"<cout<<"Thankyou!
ByeBye!
"<cout<<"*****************************************"<break;
}
}
结果:
/*******************************语法分析器源代码******************************/
#include
#include
#include
#include
chara[50],b[50],d[200],e[10];
charch;
intn1,i1=0,flag=1,n=5;
inttotal=0;
intE();
intE1();
intT();
intG();
intS();
intF();
voidinput();
voidinput1();
voidoutput();
voidmain()/*递归分析*/
{
intf,p,j=0;
charx;
d[0]='E';
d[1]='=';
d[2]='>';
d[3]='T';
d[4]='G';
d[5]='#';
printf("Pleaseinputcharacterstring(length<50,endof'#'):
\n");
do{
scanf("%c",&ch);
a[j]=ch;
j++;
}while(ch!
='#');
n1=j;
ch=b[0]=a[0];
printf("步骤\t文法\t分析串\t\t分析字符\t剩余串\n");
f=E1();
if(f==0)return;
if(ch=='#')
{printf("\nAccept!
RightExpression!
\n\n");
p=0;
x=d[p];
while(x!
='#'){
printf("%c",x);p=p+1;x=d[p];/*输出推导式*/
}
}
else{
printf("\nError\n");
printf("回车返回\n");
getchar();getchar();
return;
}
printf("\n");
printf("回车返回\n");
getchar();
getchar();
}
intE1()
{intf,t;
printf("%d\tE-->TG\t",total);total++;
flag=1;
input();
input1();
f=T();
if(f==0)return(0);
t=G();
if(t==0)return(0);
elsereturn
(1);
}
intE()
{intf,t;
printf("%d\tE-->TG\t",total);total++;
e[0]='E';e[1]='=';e[2]='>';e[3]='T';e[4]='G';e[5]='#';
output();
flag=1;
input();
input1();
f=T();
if(f==0)return(0);
t=G();
if(t==0)return(0);
elsereturn
(1);
}
intT()
{intf,t;
printf("%d\tT-->FS\t",total);total++;
e[0]='T';e[1]='=';e[2]='>';e[3]='F';e[4]='S';e[5]='#';
output();
flag=1;
input();
input1();
f=F();
if(f==0)return(0);
t=S();
if(t==0)return(0);
elsereturn
(1);
}
intG()
{intf;
if(ch=='+'){
b[i1]=ch;
printf("%d\tG-->+TG\t",total);total++;
e[0]='G';e[1]='=';e[2]='>';e[3]='+';e[4]='T';e[5]='G';e[6]='#';
output();
flag=0;
input();
input1();
ch=a[++i1];
f=T();
if(f==0)return(0);
G();
return
(1);
}
printf("%d\tG-->^\t",total);total++;
e[0]='G';e[1]='=';e[2]='>';e[3]='^';e[4]='#';
output();
flag=1;
input();input1();
return
(1);
}
intS()
{
intf,t;
if(ch=='*'){
b[i1]=ch;printf("%d\tS-->*FS\t",total);total++;
e[0]='S';e[1]='=';e[2]='>';e[3]='*';e[4]='F';e[5]='S';e[6]='#';
output();
flag=0;
input();input1();
ch=a[++i1];
f=F();
if(f==0)return(0);
t=S();
if(t==0)return(0);
elsereturn
(1);}
printf("%d\tS-->^\t",total);total++;
e[0]='S';e[1]='=';e[2]='>';e[3]='^';e[4]='#';
output();
flag=1;
a[i1]=ch;
input();
input1();
return
(1);
}
intF()
{intf;
if(ch=='('){
b[i1]=ch;printf("%d\tF-->(E)\t",total);total++;
e[0]='F';e[1]='=';e[2]='>';e[3]='(';e[4]='E';e[5]=')';e[6]='#';
output();
flag=0;
input();
input1();
ch=a[++i1];
f=E();
if(f==0)return(0);
if(ch==')'){
b[i1]=ch;printf("%d\tF-->(E)\t",total);total++;
flag=0;input();input1();
ch=a[++i1];
}
else{
printf("\nError\n");
return(0)