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

加入VIP,免费下载
 

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

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

下载须知

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

版权提示 | 免责声明

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

编译原理课程设计报告一个完整的编译器.docx

1、编译原理课程设计报告一个完整的编译器编译原理程序设计报告一个简单文法的编译器的设计与实现 专业班级 : 计算机1406班 组长姓名 : 宋世波 组长学号 : 指导教师 : 肖 桐 2016年12月设计分工 组长学号及姓名:宋世波分工:文法及数据结构设计词法分析语法分析(LL1)基于DAG的中间代码优化部分目标代码生成 组员1学号及姓名:黄润华分工:中间代码生成(LR0)部分目标代码生成 组员2学号及姓名:孙何奇分工:符号表组织部分目标代码生成摘要编译器是将便于人编写,阅读,维护的高级计算机语言翻译为计算机能解读、运行的低阶机器语言的程序。编译是从源代码(通常为高阶语言)到能直接被计算机或虚拟机

2、执行的目标代码(通常为低阶语言或机器语言)的翻译过程。一编译器的概述1.编译器的概念编译器是将便于人编写,阅读,维护的高级计算机语言翻译为计算机能解读、运行的低阶机器语言的程序。编译器将原始程序作为输入,翻译产生使用目标语言的等价程序。源代码一般为高阶语言如Pascal、C+、Java 等,而目标语言则是汇编语言或目标机器的目标代码,有时也称作机器代码。2编译器的种类编译器可以生成用来在与编译器本身所在的计算机和操作系统(平台)相同的环境下运行的目标代码,这种编译器又叫做“本地”编译器。另外,编译器也可以生成用来在其它平台上运行的目标代码,这种编译器又叫做交叉编译器。交叉编译器在生成新的硬件平

3、台时非常有用。“源码到源码编译器”是指用一种高阶语言作为输入,输出也是高阶语言的编译器。例如: 自动并行化编译器经常采用一种高阶语言作为输入,转换其中的代码,并用并行代码注释对它进行注释(如OpenMP)或者用语言构造进行注释(如FORTRAN的DOALL指令)。3.本编译器概述 编译程序的工作过程一般可以分为五个阶段:词法分析、语法分析、语义分析与中间代码产生、优化、目标代码生成。每一个阶段在功能上是相对独立的,它一方面从上一个阶段获取分析的结果来进行分析,另一方面由将结果传递给下一个阶段。由编译程序的五个阶段就对应了编译系统的结构,这五个对应阶段分为编译器的前段,中间代码以及后端。其中词法

4、分析器利用超前搜索、状态转换等方法,将源程序转化成为一个一个的单词符号二元式。一般程序语言的单词符号包括关键字、运算符、常数、标识符和界符。语法分析器将这些单词符号作为输入,对它进行语法分析。语法分析采用LL1分析法,语法分析器把语法单元作为输入供语义分析器使用。在语法分析的同时进行语法分析,并产生一定的语义动作,来生成中间代码。优化和目标代码生成是针对某一种处理器而言的。代码优化是将语义分析生成的中间代码进行优化,产生执行效率更高的代码。目标代码生成最终生成可以在某种机器上运行的机器语言或者汇编语言。还要有符号表可供查询。在整个编译过程中还包括对表格的操作和对错误的处理,这些也都是非常重要的

5、环节。环境:编译器整体全部使用visual studio2015编写目标代码在8086指令集机器上运行关键词:编译原理,前端,中间代码生成,后端,1. 概 述本程序实现的数据类型有整型(int)、浮点型(float)、char(字符型)、字符串型(string),同时在最后的目标代码生成部分允许出现布尔(bool)类型,实现的操作有if,else以及while循环,和输入输出语句,能做到直接生成目标代码并运行。做到了类型检查和重定义的判断,同时也有优化部分。词法分析部分构建得到的词法序列为二元式,二元式由两部分组成,其种别码和类型组成,其中关键字和界符记录其数字,用到时只需查其对应的关键字表和

6、界符表即可,而字符类型、字符串类型、数字类型、以及标识符类型的值一律用字符串存储,其中数字类型存储时对其使用atoi和itoa函数进行数字转字符串即可,要使用到其数字时只需要将对其使用字符串转数字函数再获得即可。语法分析部分采用的是LL1分析方法,这是一种自上而下的语法分析法,又称为预测分析法,语法分析器部分实现了自动求first集合和follow集合,采用的是递归程序获得select集合,在实现对产生式完全扫描以后,便可以获得一张完整的分析表,表中标注了是否为预测以及对应的产生式序列,而后进行语法分析,通过于获得的分析表进行比对,进行入栈出栈,匹配到相同的则认为匹配成果,当文法匹配成功,同时

7、单词也匹配成功的时候,认为语法分析完成,语法无错误,否则报错。在符号表生成部分,程序对获取的单词序列中的标识符进行再分析,确定每一个标识符的存储范围,同时从此之中获取函数的参数表,函数表部分,构建出编译器全局可用的活动记录表,以供目标代码生成使用,同时也为编译器增添新内容做准备。中间代码生成部分,在此之前需注意到,由于再语法分析部分已经生成了一个具体可理解的动作序列,所以中间代码生成部分的所用方法并非语法制导,其本身也与语法分析相割裂开来,中间代码生成采取的是自底向上进行分析,不过是基于单词序列进行的分析,其操作等于是使用已获得的伪动作序列,与源程序去作比较,找出伪动作序列的实际含义,将其顺序

8、填好,最后便完成了整个中间代码(四元式)的生成,并将其重新输出到文件中。中间代码优化部分是对填装完毕的中间代码的再处理,也就是减少无用式子,给目标代码的生成提供便利,先进行基本块划分,而后采取的是DAG图优化,即无环有向图优化,顺序是构建DAG图(无环有向图),减少无用分支,或删改部分同义分支,完成DAG图改造后便又重新由树组装四元式,组装好的四元式又再次重新输出到文件中。目标代码生成部分较长,也并不仅仅包含目标代码生成部分,在这个部分文件中,同时需要对前述获得的符号表,中间代码进行再处理,以得到符号目标代码生成所用的符号表和中间代码,再进行预处理完成之后,具体为根据四元式动作,按顺序依次生成

9、目标代码,需要依据不同的四元式动作每个采取不同的操作方法,生成相应目标代码。测试部分采用的emu8086软件,这个软件支持的指令集为8086指令集,由于64位机器上对汇编的支持并不算好,所以在8086机上最后进行的是对目标代码的正确性检验以及运行,运行通过且符合源程序实际操作即认为编译器任务完成。2. 课程设计任务及要求 设计任务1.一个简单文法的编译器前端的设计与实现 定义一个简单程序设计语言文法(包括变量说明语句、算术运算表达式、赋值语句;扩展包括逻辑运算表达式、If语句、While语句等); 扫描器设计实现; 语法分析器设计实现; 中间代码设计; 中间代码生成器设计实现。 2.难度相当的

10、自选题目, 如: 一个简单文法的编译器后端的设计与实现。 一个简单文法的编译器的设计与实现。 自选一个感兴趣的与编译原理有关的问题加以实现以下为本组选择部分:一个简单文法的编译器的设计与实现。1. 定义一个简单程序设计语言文法(包括变量说明语句、算术运算表达式、赋值语句;扩展包括逻辑运算表达式、If语句、While语句等);2. 扫描器设计实现3. 语法分析器设计实现;4. 符号表设计5. 符号表生成器设计实现6. 中间代码设计;7. 中间代码生成器设计实现。8. 中间代码优化9. 中间代码优化实现10. 目标代码设计11. 目标代码生成器设计实现 设计要求1、在深入理解编译原理基本原理的基础

11、上,对于选定的题目,以小组为单位,先确定设计方案;2、设计系统的数据结构和程序结构,设计每个模块的处理流程。要求设计合理;3、编程序实现系统,要求实现可视化的运行界面,界面应清楚地反映出系统的运行结果;4、确定测试方案,选择测试用例,对系统进行测试;5、运行系统并要通过验收,讲解运行结果,说明系统的特色和创新之处,并回答指导教师的提问;6、提交课程设计报告。以下为本组设计要求:给出一个源程序文件,作为编译器前端的输入,输出相关编译阶段的运行结果。词法分析阶段:Token序列;关键字表、界符表、符号表系统。语法分析阶段:LL1型产生式、分析表、语法分析所用栈符号表生成阶段:符号表系统中间代码生成

12、阶段:四元式序列;符号表系统。中间代码优化阶段:四元式序列、DAG图、优化完成的四元式序列目标代码生成阶段:符号表系统、四元式序列、目标代码(8086指令集)设计的文法结构产生式中文对照:1. - ( ) 2. -int|float|char|void|$3. - ( ) | | |4. - 5. - 6. - * | / | $ 7. - + | - | $ 8. - | $ 9. - |10. - = | $ 11. - 12. - , | $ 13. - 14. - | $ 15. - ;16. - | | | | $ 17. - 18. - = ; | ( ) ; 19. - 20.

13、- , | $ 21. - | | 22. -while()23. - 24. - | = |=|= 25. - if ( ) 26. - else | $ 27. - return ; 28. -cout ;|cout ;|cout ;29. -cin;产生式如下:funcdef%type&id&(¶state&)&funcblock&#type%int|float|char|void&#factor%(&exp&)|id|number|ch&#exp%divi&item&#divi%factor&faccycle&#faccycle%*&factor&faccycle|/&fact

14、or&faccycle|$&#item%+&divi&item|-&divi&item|$&#parastate%state&stateclo|$&#state%type&id&init|id&init&#init%=&rvalue|$&#rvalue%exp&#stateclo%,&stateclo|$&#funcblock%staclo&funcbloclo&#staclo%statement&staclo|$&#statement%state&;&#funcbloclo%opera&funcbloclo|whilecycle&funcbloclo|condistate&funcblocl

15、o|funcend&funcbloclo|coutstate&funcbloclo|cinstate&funcbloclo|$&#opera%id&callstate&#callstate%=&rvalue&;|(¶list&)&;&#paralist%para¶clo&#paraclo%,¶¶clo|$&#para%id|number|string&#whilecycle%while&(&logicexp&)&do&funcblock&we&#logicexp%exp&logicopera&exp&#logicopera%|=|=&#condistate%if

16、&(&logicexp&)&funcblock&nor&ie&#nor%else&funcblock&|$&#funcend%return&factor&;&#coutstate%cout&id&;&#do%$&#we%$&#ie%$&#词法分析序列表:标识符表i常数表C关键字表KIntFloatCharStringVoidIfElseSwitchCaseForDoWhileContinueBreakDefaultSizeofReturnCoutCin12345678910111213141516171819界符表P=,+,-,*,/,;,(,) ,;/*界符表*/char variate16

17、10 = ;/*变量表*/static struct twoele /*二元式数据结构*/ style kind; char value125; int value2; 二元式包含三项,但实际使用时只会用到其中两项,单词种类的不同决定了其对应有不同的处理方式,kind用来存放种类,value1和value2都是用来存储单词所含值的,相对于不同类单词有不同处理方式。而其中kind所可为的值以及在上面数据结构中相应的有所列出的,对应查较即可。具体分配如下,当单词为关键字或者界符时,使用的是kind和value2项,即value2存储对应固定表中单词所对序列。而当单词为其他时,即单词为字符、字符串、

18、标识符或者数字时,采取将单词直接字符串化直接存储其值,为数字时调用库函数将其字符化,取出时候按照其相应种别码采取不同的取出方式。static twoele TOKEN1000;/*词法序列(二元式结构)*/ 流程图 算法static void init_twoele(twoele*tok) /*初始化二元式,同时将变量表初始化*/int tra(char*cmp) /*查询获得单词是否为关键字,是则返回关键字序号*/void addvar(char*cmp) /*加入变量表*/void prep(char ch) /*预处理,过滤注释*/static void scanner() /*词法扫描

19、*/ 在词法分析之前有一段的注释处理部分,其相应方案是在遇到/符时即进入注释处理,而后再次遇到/符时候认为离开注释处理部分,但有所控制,若较长范围内并未再遇到/符则跳出自动机,认为/符号就是一个单纯的界符,由于c语言中并没有/所用的语言,如果此时并不在字符串中,可直接进行报错处理。以下就词法分析遇到的单词处理方法进行分情况解释:字符:首先是用if判断字符,字符都会带单引号,所以若识别出单引号则进入字符判别部分,执行ch=fgetc(fp),进入下一个判断语句,若条件为假则输出error,这样可以完成一个简单的报错功能。若为真的话,执行ch=fgetc(fp),判断是否为单引号,若为假则输出er

20、ror,为真就要将找到ch对应的token。(这一段包括在界符处理的字部分中)字符串:字符串的判断与字符的判断差不多,不同的是字符串要识别的是双引号,而且需要加一个循环来将整个字符串识别完毕。首先ch若为“,则进入字符串的判别部分,ch=fgetc(fp),然后就要进入一个while循环,每循环一次就执行ch=fgetc(fp),若为数字或字母就将其装入字符数组,否则跳出循环。剩下的就同识别字符一样,若识别不到”,就输出error。然后查找token和压队列的操作。标识符:由于关键字也是一些字符串,所以一般放在一起来判断,若不是关键字则为标识符。由于关键字和标识符都必须是以字母开头,所以用if

21、为字符来判断,为真接着判断,while循环来识别完整个标识符或关键字的部分与识别字符串相同,也是装入字符数组,但比字符串部分多出一点就是标识符可以有下划线,所以中间判断条件会变成数字字符下划线均可。While循环结束后需要在字符数组的最后一位后加上0。识别完毕后就需要判断该字符串到底是标识符还是关键字了,关键字个数是固定的,挨个匹配就可以了,界符:判断方法简单粗暴,直接switch即可,不过要注意比如=这种需要多判一次,即单词读取应向后延伸一位,其他并没有什么难题,需要注意的是,字符以及字符串判断也放入了这里面作字部分(由于字符和字符串开头的单引号和“双引号也算是实际意义上的界符)数字:最后是

22、数字的识别,首先,识别数字直接进行if比较,接着同字符串一样进入循环识别整个数字,都存入整形数组。由于要存入常数表的应该是一个数而不是整形数组,所以应做出一个变量用于计算置位,大致思想就是用循环来将数组中的数乘10,数组的第一位先乘10,每循环一次增加数组下一位。将整数部分都识别完了后就要判断是否为小数或者科学计数法,若ch为小数点就进入小数的判断部分,小数部分从数组转变为小数的程序就是将上述的函数中的乘便为除,然后再识别符号后面的数,转化为整数,将之前识别出来的数除或乘该整数次数的10,而后用数字转字符串函数存入二元式中,二元式种类写一个常数种别码。 运行截图 语法分析器模块(负责人:宋世波

23、) 功能1.获取产生式2.自动求first和follow集合3.构建分析表4.构建分析栈5.通过查分析表进行语法分析6.输出动作序列LL1文法解释:LL(1)分析法是指从左到右扫描、最左推导(LL)和只查看一个当前符号(括号中的 1)之意; LL(1)分析法又称预测分析法,与递归子程序法同属于自顶向下确定性语法分析方法; LL(1) 分析法的基本要点有三:1 利用一个分析表,登记如何选择产生式的知识;2 利用一个分析栈,记录分析过程;3 此分析法要求文法必须是 LL(1)文法。语法分析是编译中的第二阶段,它的任务是识别和处理比单词更大的语法单位,判断源程序在结构上是否正确。 从形式上来说,语法

24、分析是对一个给定的字符串,判断它是否为文法的一个句子。在我设计的语法分析器中,直接采用LL(1)分析方法。当然由于本身文法为上下文无关文法,所以纯LL1并没有使用问题,此外,对于错误的识别,该语法分析器可以输出错误信息。 数据结构struct pronode /*产生式节点*/ type it; char symbol15;struct product /*产生式数据结构*/ char vn15; pronode itself10;product c100;产生式是从文件中读入的,也就是本语言的文法结构,其中产生式结构的首点product为产生式左部,pronode部分为右部。与实际产生式进行比较,很容易知道这个数据结构所允许的任何产生式右部最多能容纳十个元素,左部元素长度不超过十五。对于文法中的终结符和非终结符,在递归子程序方法中非终结符直接转换为字符串,终结符则是对应的Token,但当调用LL(1)子程序时,需要将非终结符和终结符都转换为string类型,这样做的好处是统一格式之后便于进行压栈弹栈等操作,但是会提高算法的时间复杂度,属于以时间换空间

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

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