编译原理语义分析和中间代码生成精选文档.ppt
《编译原理语义分析和中间代码生成精选文档.ppt》由会员分享,可在线阅读,更多相关《编译原理语义分析和中间代码生成精选文档.ppt(169页珍藏版)》请在淘文阁 - 分享文档赚钱的网站上搜索。
1、编译原理语义分析和编译原理语义分析和中间代码生成中间代码生成本讲稿第一页,共一百六十九页4.1 概概 述述 4.1.1语义分析的概念一个源程序经过词法分析、语法分析之后,表明该源程序在书写上是正确的,并且符合程序语言所规定的语法。但是语法分析并未对程序内部的逻辑含义加以分析,因此编译程序接下来的工作是语义分析,即审查每个语法成分的静态语义。如果静态语义正确,则生成与该语言成分等效的中间代码,或者直接生成目标代码。本讲稿第二页,共一百六十九页直接生成机器语言或汇编语言形式的目标代码的优点是编译时间短且无需中间代码到目标代码的翻译,而中间代码的优点是使编译结构在逻辑上更为简单明确,特别是使目标代码
2、的优化比较容易实现。本讲稿第三页,共一百六十九页如同在进行词法分析、语法分析的同时也进行着词法检查、语法检查一样,在语义分析时也必然要进行语义检查。动态语义检查需要生成相应的目标代码,它是在运行时进行的;静态语义检查是在编译时完成的,它涉及以下几个方面:(1)类型检查,如参与运算的操作数其类型应相容。(2)控制流检查,用以保证控制语句有合法的转向点。如C语言中不允许goto语句转入case语句流;break语句需寻找包含它的最小switch、while或for语句方可找到转向点,否则出错。本讲稿第四页,共一百六十九页(3)一致性检查,如在相同作用域中标识符只能说明一次、case语句的标号不能相
3、同等。语义分析阶段只产生中间代码而不生成目标代码的方法使编译程序的开发变得较为容易,但语义分析不像词法分析和语法分析那样可以分别用正规文法和上下文无关文法描述。由于语义是上下文有关的,因此语义的形式化描述是非常困难的,目前较为常见的是用属性文法作为描述程序语言语义的工具,并采用语法制导翻译的方法完成对语法成分的翻译工作。本讲稿第五页,共一百六十九页4.1.2语法制导翻译方法语法制导翻译的方法就是为每个产生式配上一个翻译子程序(称语义动作或语义子程序),并在语法分析的同时执行这些子程序。语义动作是为产生式赋予具体意义的手段,它一方面指出了一个产生式所产生的符号串的意义,另一方面又按照这种意义规定
4、了生成某种中间代码应做哪些基本动作。在语法分析过程中,当一个产生式获得匹配(对于自上而下分析)或用于归约(对于自下而上分析)时,此产生式相应的语义子程序就进入工作,完成既定的翻译任务。本讲稿第六页,共一百六十九页语法制导翻译分为自下而上语法制导翻译和自上而下语法制导翻译,我们重点介绍自下而上语法制导翻译。假定有一个自下而上的LR分析器,我们可以把这个LR分析器的能力加以扩大,使它能在用某个产生式进行归约的同时调用相应的语义子程序进行有关的翻译工作;每个产生式的语义子程序执行之后,某些结果(语义信息)必须作为此产生式的左部符号的语义值暂时保存下来,以便以后语义子程序引用这些信息。本讲稿第七页,共
5、一百六十九页此外,原LR分析器的分析栈也加以扩充,以便能够存放与文法符号相对应的语义值。这样,分析栈可以存放三类信息:分析状态、文法符号及文法符号对应的语义值。扩充后的分析栈如图41所示。作为一个例子,我们考虑下面的文法及语义动作所执行的程序:本讲稿第八页,共一百六十九页产生式语义动作(0)SEprintvalTOP(1)EE(1)+E(2)valTOP=valTOP+valTOP+2(2)EE(1)*E(2)valTOP=valTOP*valTOP+2(3)E(E(1)valTOP=valTOP+1(4)EivalTOP=lexval(注:lexval为i的整型内部值)这个文法的LR分析表见
6、表3.20。本讲稿第九页,共一百六十九页我们扩充分析栈工作的总控程序功能,使其在完成语法分析的同时也能完成语义分析工作(这时的语法分析栈已成为语义分析栈);即在用某一个规则进行归约之后,调用相应的语义子程序完成与所用产生式相应的语义动作,并将每次工作后的语义值保存在扩充后的“语义值”栈中。图42表示算术表达式79*5#的语法树及各结点值,而表4.1则给出了根据表3.20用LR语法制导翻译方法得到的该表达式的语义分析和计值过程。本讲稿第十页,共一百六十九页图41扩充后的LR分析栈本讲稿第十一页,共一百六十九页图42语法制导翻译计算表达式79*5#的语法树本讲稿第十二页,共一百六十九页表4.1表达
7、式79*5#的语义分析和计值过程步骤状态栈符号栈语义栈输入串主要动作10#_79*5#s3203#7_9*5#r4301#E_79*5#s44014#E+_7_9*5#s350143#E+9_7_*5#r460147#E+E_7_9*5#s5701475#E+E*_7_9_5#s38014753#E+E*5_7_9_#r49014758#E+E*E_7_9_5#r2100147#E+E_7_45#r11101#E_52#acc本讲稿第十三页,共一百六十九页4.2 属属 性性 文文 法法4.2.1文法的属性属性是指与文法符号的类型和值等有关的一些信息,在编译中用属性描述处理对象的特征。随着编译的
8、进展,对语法分析产生的语法树进行语义分析,且分析的结果用中间代码描述出来。对于一棵等待翻译的语法树,它的各个结点都是文法中的一个符号X,该X可以是终结符或非终结符。根据语义处理的需要,在用产生式AX进行归约或推导时,应能准确而恰当地表达文法符号X在归约或推导时的不同特征。本讲稿第十四页,共一百六十九页例如,判断变量X的类型是否匹配,要用X的数据类型来描述;判断变量X是否存在,要用X的存储位置来描述;而对X的运算,则要用X的值来描述;因此,语义分析阶段引入X的属性,如X.type、X.place、X.val等来分别描述变量X的类型、存储位置以及值等不同的特征。文法符号的属性可分为继承属性与综合属
9、性两类。继承属性用于“自上而下”传递信息。继承属性由相应语法树中结点的父结点属性计算得到,即沿语法树向下传递,由根结点到分枝(子)结点,它反映了对上下文依赖的特性。继承属性可以很方便地用来表示程序语言上下文的结构关系。本讲稿第十五页,共一百六十九页综合属性用于“自下而上”传递信息。综合属性由相应语法分析树中结点的分枝结点(即子结点)属性计算得到,其传递方向与继承属性相反,即沿语法分析树向上传递,从分枝结点到根结点。本讲稿第十六页,共一百六十九页4.2.2属性文法属性文法是一种适用于定义语义的特殊文法,即在语言的文法中增加了属性的文法,它将文法符号的语义以“属性”的形式附加到各个文法的符号上(如
10、上述与变量X相关联的属性X.type、X.place和X.val等),再根据产生式所包含的含义,给出每个文法符号属性的求值规则,从而形成一种带有语义属性的上下文无关文法,即属性文法。属性文法也是一种翻译文法,属性有助于更详细地指定文法中的代码生成动作。本讲稿第十七页,共一百六十九页例如,简单算术表达式求值的属性文法如下:产生式语义规则(1)SEprint(E.val)(2)EE(1)+TE.val=E(1).val+T.val(3)ETE.val=T.val(4)TT(1)*F T.val=T(1).val*F.val(5)TT(1)T.val=T(1).val(6)F(E)F.val=E.v
11、al(7)FiF.val=i.lexval本讲稿第十八页,共一百六十九页上面的一组产生式中,每一个非终结符都有一个属性val来表示整型值,如E.val表示E的整型值,而i.lexval则表示i的整型内部值。与产生式关联的每一个语义规则的左部符号E、T、F等的属性值的计算由其各自相应的右部符号决定,这种属性也称为综合属性。与产生式SE关联的语义规则是一个函数print(E.val),其功能是打印E产生式的值。S在语义规则中没有出现,可以理解为其属性是一个虚属性。我们再举一例说明属性文法。一简单变量类型说明的文法GD如下:GD:DintLfloatLLL,idid本讲稿第十九页,共一百六十九页其对
12、应的属性文法为:产生式语义规则(1)DTLL.in=T.type(2)TintT.type=int(3)TfloatT.type=float(4)LL(1),idL(1).in=L.in;addtype(id.entry,L.in)(5)Lidaddtype(id.entry,L.in)注意到与文法GD相应的说明语句形式可为intid1,id2,,idn或者floatid1,id2,,idn本讲稿第二十页,共一百六十九页非终结符T有一个综合属性type,其值为int或float。语义规则L.in=T.type表示L.in的属性值由相应说明语句指定的类型T.type决定;属性L.in被确定后将随
13、语法树的逐步生成而传递到下边的有关结点使用,这种结点属性称为继承属性。由此可见,标识符的类型可以通过继承属性的复写规则来传递。例如,对输入串inta,b,根据上述的语义规则,可在其生成的语法树中看到用“”表示的属性传递情况,如图43所示。本讲稿第二十一页,共一百六十九页图43属性信息传递情况示意本讲稿第二十二页,共一百六十九页4.3 几种常见的中间语言几种常见的中间语言4.3.1抽象语法树抽象语法树也称图表示,是一种较为流行的中间语言表示形式。在抽象语法树表示中,每一个叶结点都表示诸如常量或变量这样的运算对象,而其它内部结点则表示运算符。抽象语法树不同于前述的语法树,它展示了一个操作过程并同时
14、描述了源程序的层次结构。本讲稿第二十三页,共一百六十九页注意,语法规则中包含的某些符号可能起标点符号作用也可能起解释作用。如赋值语句语法规则:SV=e其中的赋值号“=”仅起标点符号作用,其目的是把V与e分开;而条件语句语法规则:Sif(e)S1;elseS2本讲稿第二十四页,共一百六十九页其中的保留字符号if和else起注释作用,说明当布尔表达式e为真时执行S1,否则执行S2;而“;”仅起标点符号作用。可以看出,上述语句的本质部分是V、e和Si。当把语法规则中本质部分抽象出来而将非本质部分去掉后,便得到抽象语法规则。这种去掉不必要信息的做法可以获得高效的源程序中间表示。上述语句的抽象语法规则为
15、:本讲稿第二十五页,共一百六十九页(1)赋值语句:左部表达式(2)条件语句:表达式语句1语句2与抽象语法相对应的语法树称为抽象语法树或抽象树,如赋值语句x=ab*c的抽象语法树如图44(a)所示,而图44(b)则是该赋值语句的普通语法树。本讲稿第二十六页,共一百六十九页图44x=ab*c的语法树本讲稿第二十七页,共一百六十九页抽象语法树的一个显著特点是结构紧凑,容易构造且结点数较少。图44(b)所示的普通语法树的结点为14个;而图44(a)所示的抽象语法树的结点仅有7个,且每个内部结点最多只有两个分支,因此可以将每个赋值语句或表达式表示为一棵二叉树。对于含有多元运算的更为复杂的语法成分,相应的
16、抽象语法树则为一棵多叉树,但我们总可以将其转变为一棵二叉树。本讲稿第二十八页,共一百六十九页4.3.2逆波兰表示法逆波兰表示法是波兰逻辑学家卢卡西维奇(Lukasiewicz)发明的一种表示表达式的方法,这种表示法把运算量(操作数)写在前面,把运算符写在后面,因而又称后缀表示法。例如,把a+b写成ab+,把a*(b+c)写成abc+*。本讲稿第二十九页,共一百六十九页1表达式的逆波兰表示表达式E的后缀表示的递归定义如下:(1)如果E是变量或常数,则E的后缀表示即E自身。(2)如果E为E1opE2形式,则它的后缀表示为E1E2op;其中op是二元运算符,而E1、E2分别又是E1和E2的后缀表示。
17、若op为一元运算符,则视E1和E1为空。(3)如果E为(E1)形式,则E1的后缀表示即为E的后缀表示。本讲稿第三十页,共一百六十九页上述递归定义的实质是:后缀表示中,操作数出现的顺序与原来一致,而运算符则按运算先后的顺序放入相应的操作数之后(即运算符先后的顺序发生了变化)。这种表示已不再需要用括号来规定运算的顺序了。后缀表示中的计值用栈实现非常方便。一般的计值过程是自左至右扫描后缀表达式,每碰到运算量就把它推进栈,每碰到K目运算符就把它作用于栈顶的K个运算量,并用运算的结果(即一个运算量)来取代栈顶的K个运算量。本讲稿第三十一页,共一百六十九页2程序语句的逆波兰表示为了用逆波兰式表示一些控制语
18、句,我们定义转移操作如下:(1)BL:转向某标号;(2)BT:条件为真时转移;(3)BF:条件为假时转移;(4)BR:无条件转移。本讲稿第三十二页,共一百六十九页部分程序语句的逆波兰表示如下:(1)赋值语句。赋值语句“=”的逆波兰表示为=例如,赋值语句“x=a+b*c”可按逆波兰式写为“xabc*+=”。(2)GOTO语句。转向语句“GOTO”的逆波兰表示为BL其中,“BL”为单目后缀运算符,“”则为BL的一个运算分量。本讲稿第三十三页,共一百六十九页(3)条件语句。BR表示无条件转移单目后缀运算符。例如,“BR”表示无条件转移到“”处,这里的顺序号是BR的一个特殊运算分量,用来表示逆波兰式中
19、单词符号的顺序号(即第几个单词),它不同于GOTO语句中的语句标号。BT和BF表示按条件转移的两个双目后缀运算符。例如:BTBF本讲稿第三十四页,共一百六十九页分别表示当e为真或假时转移到顺序号处;其中,布尔表达式e的逆波兰式和顺序号是两个特殊的运算分量。若 使 用 BT和 BF两 个 运 算 符,则 条 件 语 句if(e)S1;elseS2的逆波兰式为:BF/*e为假则转S2的第一个单词的顺序号*/*e为真则执行S1*/BR/*S1执行结束后无条件转出该条件语句*/本讲稿第三十五页,共一百六十九页(3)条件语句。BR表示无条件转移单目后缀运算符。例如,“BR”表示无条件转移到“”处,这里的
20、顺序号是BR的一个特殊运算分量,用来表示逆波兰式中单词符号的顺序号(即第几个单词),它不同于GOTO语句中的语句标号。BT和BF表示按条件转移的两个双目后缀运算符。例如:BTBF本讲稿第三十六页,共一百六十九页例如,条件语句if(mn)k=i+1;elsek=i1的逆波兰式表示为(1)(18)为单词编号):(1)mn(4)13BF(6)ki1+=(11)18BR(13)ki1=(18)if语句的后继语句此逆波兰式也可写在一行上,即mn13BFki1+=18BRki1=本讲稿第三十七页,共一百六十九页(4)循环语句。for循环语句为:for(i=m;i=n;i+)S;其中,i为循环控制变量,m为
21、初值,n为终值,S为循环体。循环语句不能直接用逆波兰表示,因而将其展开为等价的条件语句后再用逆波兰表示,即i=m;10:if(i=n)S;i=i+1;gotol0本讲稿第三十八页,共一百六十九页4.3.3三地址代码1三地址代码的形式三地址代码语句的一般形式为x=yopz其中,x、y和z为名字、常量或编译时产生的临时变量;op为运算符,如定点运算符、浮点算符和逻辑运算符等。三地址代码的每条语句通常包含三个地址,两个用来存放运算对象,本讲稿第三十九页,共一百六十九页一个用来存放运算结果。在实际实现中,用户定义的名字将由指向符号表中该名字项的指针所取代。由于三地址语句只含有一个运算符,因此多个运算符
22、组成的表达式必须用三地址语句序列来表示,如表达式x+y*z的三地址代码为:t1=y*zt2=x+t1本讲稿第四十页,共一百六十九页其中,t1和t2是编译时产生的临时变量。三地址代码是语法树的一种线性表示,如图44(a)所示的语法树用三地址代码表示为:t1=b*ct2=at1x=t2本讲稿第四十一页,共一百六十九页2三地址语句的种类作为中间语言的三地址语句非常类似于汇编代码,它可以有符号标号和各种控制流语句。常用的三地址语句有以下几种:(1)x=yopz形式的赋值语句,其中op为二目的算术运算符或逻辑运算符。(2)x=opy形式的赋值语句,其中op为一目运算符,如一目减uminus、逻辑否定no
23、t、移位运算符以及将定点数转换成浮点数的类型转换符。本讲稿第四十二页,共一百六十九页(3)x=y形式的赋值语句,将y的值赋给。(4)无条件转移语句gotoL,即下一个将被执行的语句是标号为L的语句。(5)条件转移语句ifxropygotoL,其中rop为关系运算符,如、=等。若x和y满足关系rop就转去执行标号为L的语句,否则继续按顺序执行本语句的下一条语句。本讲稿第四十三页,共一百六十九页(6)过程调用语句parX和callP,n。源程序中的过程调用语句P(X1、X2、,Xn)可用下列三地址代码表示:parX1parX2parXncallP,n其中,整数n为实参个数。过程返回语句为retur
24、ny,其中y为过程返回值。本讲稿第四十四页,共一百六十九页(7)变址赋值语句x=yi,其中x、y、i均代表数据对象,表示把从地址y开始的第i个地址单元中的值赋给x。xi=y则表示把y的值赋给从地址x开始的第i个地址单元。(8)地址和指针赋值语句x=&y表示将y的地址赋给x,y可以是一个名字或一个临时变量,而x是指针名或临时变量;x=*y表示将y所指示的地址单元中的内容(值)赋给x,y是一个指针或临时变量;*x=y表示指将x所指对象的值置为y的值。本讲稿第四十五页,共一百六十九页3三地址代码的具体实现三地址代码是中间代码的一种抽象形式。在编译程序中,三地址代码语言的具体实现通常有三种表示方法:四
25、元式、三元式和间接三元式。1)四元式四元式是具有四个域的记录结构,这四个域为(op,arg1,arg2,result)本讲稿第四十六页,共一百六十九页其中,op为运算符;arg1、arg2及result为指针,它们可指向有关名字在符号表中的登记项或一临时变量(也可空缺)。常用的三地址语句与相应的四元式对应如下:x=yopz对应(op,y,z,x)x=y对应(uminus,y,_,x)x=y对应(=,y,_,x)parx1对应(par,x1,_,_)callP对应(call,_,_,P)gotoL对应(j,_,_,L)ifxropygotoL对应(jrop,x,y,L)本讲稿第四十七页,共一百六
- 配套讲稿:
如PPT文件的首页显示word图标,表示该PPT已包含配套word讲稿。双击word图标可打开word文档。
- 特殊限制:
部分文档作品中含有的国旗、国徽等图片,仅作为作品整体效果示例展示,禁止商用。设计者仅对作品中独创性部分享有著作权。
- 关 键 词:
- 编译 原理 语义 分析 中间 代码 生成 精选 文档
限制150内