小言_互联网的博客

编译器做了什么

339人阅读  评论(0)


从最直观的角度来讲,编译器就是将高级语言翻译成机器语言的一个工具,比如我们用C/C++语言写的一个程序可以使用编译器将其翻译成机器可以执行的指令及数据。使用机器指令或汇编语言编写程序是十分费事及乏味的事情,它们使得程序开发的效率十分低下,并且使用机器语言或汇编语言编写的程序依赖于特定的机器,一个为某种CPU编写的程序在另外一种CPU下完全无法运行,要重新编写,这几乎是令人无法接受的。所以人们期望能够采用类似于自然语言的语言来描述一个程序,但是自然语言的形式不够精确,所以类似于数学定义的编程语言很快就诞生了,高级语言使得程序员们能够更好关注程序逻辑本身,而尽量少考虑计算机本身的限制,如字长,内存大小,通信方式,存储方式等,高级编程语言的出现使得程序开发效率大大提高,高级语言的可移植性也使得它在多种计算机平台下能够游刃有余,据研究,高级语言的开发效率是汇编语言和机器语言的5倍以上。
编译器过程一般分为6步:扫描,语法分析,语义分析,源代码优化,代码生成和目标优码优化。如图所示:


我们将结合图来简单描述从源代码到最终目标代码的过程,以一段很简单的C语言的代码为例子来讲述这个过程,比如我们有一行C语言的源代码如下:

array[index] =(index+4)*(2+6)
CompilerExpression.c

词法分析

首先源代码程序被输入到扫描器,扫描器的任务很简单,它只是简单地进行词法分析,运用一种类似于有限状态机的算法可以很轻松地将源代码的字符序列分割成一系列的记号,比如上面的那行程序,总共包含了28个非空字符,经过扫描后,产生了16个记号

记号 类型
array 标识符
[ 左方括号
index 标识符
] 右方括号
= 赋值
左圆括号
index 标识符
+ 加号
4 数字
右圆括号
* 乘号
左圆括号
2 数字
+ 加号
6 数字
右圆括号

词法分析产生的记号一般可以分为如下几类:关键字,标识符,字面量(包括数字,字符串等)和特殊符号(如加号,等号)。在识别记号的同时,扫描器也完成了其他工作,比如将标识符存放到符号表,将数字,字符串常量存放到文字表等,以备后面的步骤使用。
有一个叫做lex的程序可以实现词法扫描,它会按照用户之前描述好的词法规则将输入的字符串分割成一个个记号,因为这样一个程序的存在,编译器的开发者就无须为每一个编译器开发一个独立的词法扫描器,而是根据需要改变词法规则就可以了。
另外对于一些有预处理的语言,比如C语言,它的宏替换和文件包含等工作一般不归入编译器的范围而交给一个独立的预处理器。

语法分析

接下来语法分析器将对由扫描器产生的记号进行语法分析,从而产生语法树,整个分析过程采用了上下无关语法的分析手段,简单来讲,由语法分析器生成的语法树就是以表达式为节点的树,我们知道,C语言的一个语句是一个表达式,而复杂的语句是很多表达式的组合。上述例子中的语句就是一个由赋值表达式,加法表达式,乘法表达式,数组表达式,括号表达式组成的复杂语法。它在经过语法分析器以后形成如图所示的语法树。

从图中可以看出,整个语句被看作是一个赋值表达式,赋值表达式的左边是一个数组表达式,它的右边是一个乘法表达式,数组表达式又由两个符号表达式组成,等等。符号和数字是最小的表达式,它们不是由其他的表达式来组成的,所以它们通常作为整个语法树的叶节点。在语法分析的同时,很多运算符号的优先级和含义也被确定下来了,比如乘法表达式的优先级比加法高,而圆括号表达式的优先级比乘法高,等等。另外有些符号具有多重含义,比如星号*在C语言中可以表示乘法表达式,也可以表示对指针取内容的表达式,所以语法分析阶段必须对这些内容进行区分。如果出现了表达式不合法,比如各种括号不匹配,表达式中缺少操作符等,编译器就会报告语法分析阶段的错误。
语法分析也有一个现成的工具叫做yacc。它也像lex一样,可以根据用户给定的语法规则对输入的记号序列进行解析,从而构建出一颗语法树,对于不同的编程语言,编译器的开发者只须改变语法规则,而无需为每一个编译器编写一个语法分析器,所以它被称为“编译器编译器”。

语义分析

接下来进行的是语义分析,由语义分析器来完成,语法分析仅仅是完成了对表达式的语法层面的分析,但是它并不了解这个语句是否真正有意义。比如C语言里面两个指针做乘法运算是没有意义的,但是这个语句在语法上是合法的,比如同样一个指针和一个浮点数做乘法运算是否合法等,编译器所能分析的语义是静态语义,所谓静态语义是指在编译期可以确定的语义,与之对应的动态语义就是只有在运行期才能确定的语义。
静态语义通常包括声明和类型的匹配,类型的转换。比如当一个浮点型的表达式赋值给一个整型的表达式时,其中隐含了一个浮点型到整型转换的过程,语义分析过程中需要这个步骤。比如将一个浮点型赋值给一个指针的时候,语义分析程序就会发现这个类型不匹配,编译器就会报错。
动态语义一般指在运行期出现的语义相关的问题,比如将0作为除数是一个运行期语义错误。
经过语义分析阶段以后,整个语法树的表达式都被标识了类型,如果有些类型需要做隐式转换,语义分析程序会在语法树中插入相应的转换节点。上面描述的语法树在经过语义分析阶段以后成为如图的形式。

可以看到,每个表达式(包括符号和数字)都被标识了类型,我们的例子中几乎所有的表达式都是整型的,所有无须做转换做转换,整个分析过程很顺利。语义分析器还对符号表里的符号类型也做了更新。

中间原理生成

现代的编译器有着很多层次的优化,往往在源代码级别会有一个优化过程,我们这里所描述的源码级优化器在不同编译器中可能会有不同的定义或有一些其他的差异,源代码优化器会在源代码级别进行优化,在上例中,(2+6)这个表达式可以被优化掉,因为它的值在编译期就可以被确定。经过优化的语法树如图所示。

我们看到(2+6)这个表达式被优化成8.其实直接在语法树上作优化很难,所以源代码优化器往往将整个语法树转换为中间代码,它是语法树的顺序表达,其实它已经很接近目标代码,它一般跟目标机器和运行时环境是无关的,比如它不包括数据的尺寸,变量地址和寄存器的名字等,中间代码有很多种类型,在不同的编译器中有着不同的形式,比较常见的有:三地址码和p-代码。
最基础的三地址码是这样的:
x=y op z
这个三地址码表示将变量y和z进行op操作以后,赋值给x。这里op操作可以是算数运算,比如加减乘除等,也可以是其他任何可以应用到y和z的操作。三地址码也得名于此。上面例子可以被翻译成如下代码:

t1=2+6;
t2=index+4
t3=t2*t1
array[index]=t3

我们可以看到,为了使所有的操作都符合三地址码形式,这里利用了几个临时变量t2和t3.在三地址码的基础上进行优化时,优化程序会将2+6的结果计算出来,得到t1=8,然后将后面代码中的t1替换成数字8.还可以省去一个临时变量t3,因为t2可以重新利用。经过优化以后的代码如下:

t2=index+4
t2=t2*8
array[index]=t2

中间代码使得编译器可以被分为前端和后端,编译器前端负责产生机器无关的中间代码,编译器后端将中间代码转换成目标机器代码。这样对于一些可以跨平台的编译器而言,它们可以针对不同的平台使用同一个前端和针对不同机器平台的数个后端。

目标代码生成和优化

源代码优化器产生中间代码标准着下面的过程都属于编译器后端。编译器后端主要包括代码生成器和目标代码优化器。代码生成器将中间代码转换成目标机器代码,这个过程十分依赖于目标机器,因为不同的机器有着不同的字长,寄存器,整数数据类型和浮点数数据类型等,对于上面的例子中的中间代码,代码生成器可能会生成下面的代码序列(使用x86的汇编语言来表示):

movl index,%ecx;value of index to ecx
addl $4,%ecx;ecx=ecx+4
mull $8,%ecx;value of index to eax
movl index,%eax;value of index to eax
movl %ecx,array(,eax,4);array[index]=ecx

最后目标代码优化器对上述的目标代码进行优化,比如寻找合适的寻址方式,使用位移来代替乘法运算,删除多余的指令等,上面的例子中,乘法由一条相对复杂的基址比例变址寻址的lea指令完成,随后由一条mov指令完成最后的赋值操作。

movl index,%edx
leal 32(,%edx,8),%eax
movl %eax,array(,%edx,4)

现代的编译器有着异常复杂的结构,这是因为现代高级编程语言本身非常地复杂,比如C++语言的定义就极为复杂,至今没有一个编译器能够完整支持C++语言标准所规定的所有语言特性。另外现代的计算机CPU相当地复杂,CPU本身采用了诸如流水线,多发射,超标量等诸多复杂的特性,为了支持这些特性,编译器的机器指令优化过程也变得十分复杂。使得编译过程更为复杂的是有些编译器支持多种硬件平台,即允许编译器编译出多种目标CPU的代码。比如著名的GCC编译器就几乎支持所有CPU平台,这也导致了编译器的指令生成过程更为复杂。


转载:https://blog.csdn.net/m0_51174487/article/details/116068061
查看评论
* 以上用户言论只代表其个人观点,不代表本网站的观点或立场