编译器的过程,从源文件到.exe文件经过就几步,是如何进行的

从一个.cpp文件到一个exe会经过如下几步

1.预处理

2.编译

3.汇编

4.链接

1:预处理阶段有预处理器进行,会将每一个.cpp(源文件(c语言程序和c++语言程序是一样的)预处理器会将所有源文件中的与预处       理指令进行处理,所谓的预处理执行就是  #开头的语句

      如#define    #include   #if 1  #endif     宏定义,头文件包含   条件偏移等等都是预处理指令

      #pragma once(加上头文件的开头,能够保证这个头文件的内容只被编译一次)上述等等都是最常见的预处理指令

vs可以配置预处理器生成预处理阶段生成的预处理文件.i文件

编译器的过程,从源文件到.exe文件经过就几步,是如何进行的

 设置好后点重新生成,可以看到给两个源文件都生成了.i文化

这是我Main.cpp文件中的内容

#include"taolaoda.h"
#include<stdio.h>
#define  LOG(x)  printf("%d\n",x)  
int main() {
    LOG(add(1, 2));
    LOG(sub(10, 20));
    LOG(mul(3, 5));
    LOG(div(3, 6));
    getchar();
}

这是我taolaoda.cpp文件中内容

int  add(int a, int b) { return a + b; }
int  sub(int a, int b) { return  a - b; }
int  mul(int a, int b) { return a * b; }
int  div(int a, int b) {
    if (0 == b) {
        return -1;
    }
    return a / b;
}

这是我们taolaoda.h我们的内容

#pragma once
int  add(int a,int b);
int  sub(int a,int b);
int  mul(int a, int b);
int  div(int a, int b);
 

编译器的过程,从源文件到.exe文件经过就几步,是如何进行的

打开Main.i,可以看到一开始就是把taolaoda.h文件中的内容复制到Main.i中了

这个文件有1万多行,因为#include<stdio.h>,可以看到后面本来源文件使用的是宏,全都替换成了定义的

所以#defien宏定义的效率为什么高,因为它在预处理阶段就替换了不会影响执行使其的效率

而且实现一个简单的功能如何封装成宏函数,而不是普通函数,也能实现模块化,而且不用给函数开辟堆栈,效率会更高

 

编译器的过程,从源文件到.exe文件经过就几步,是如何进行的

编译器的过程,从源文件到.exe文件经过就几步,是如何进行的

taolaoda.i文件几乎没有什么变化,因为根本没有taolaoda.cpp里面没有使用预处理指令

编译器的过程,从源文件到.exe文件经过就几步,是如何进行的

 

我将taolaoda.cpp中加一个条件偏移,和一个#Include包含taolaoda.h看看  在重新生成

编译器的过程,从源文件到.exe文件经过就几步,是如何进行的

 

编译器的过程,从源文件到.exe文件经过就几步,是如何进行的

可以看到taolaoda.i文件  现在我们可以更加清楚的知道#include包含只不过是文件的复制嘛,把某个文件复制到源文件中

对了设置成了预处理文件就编译不了必须设置回去才能编译

如果最后要调试运行,需要把上述的改动 改回去,不然生成不了obj文件的

2:汇编将是将预处理生成的.i文件作为输入,生成.asm汇编文件作为输出

编译器的过程,从源文件到.exe文件经过就几步,是如何进行的

编译器的过程,从源文件到.exe文件经过就几步,是如何进行的

编译器的过程,从源文件到.exe文件经过就几步,是如何进行的

编译器的过程,从源文件到.exe文件经过就几步,是如何进行的

可能有的人有些奇怪汇编文件不是.s为什么是.asm   linux下是.s Windows下是.asm

3:汇编就是将生产的.asm(汇编)作为输入 .obj文件作为输出,我们知道CPU只能是被二进制,汇编程序作为机器指令的助记符,只是给人看的,CPU识别不了,CPU只能是被二进制,汇编就是将汇编文件转换成二进制文件.obj  也称为目标文件

4:链接将多个.obj文件作为输入生成一个exe文化可执行文件作为输出

比如我们main函数所在的.obj文件上述了是需要使用taolaoda.obj文件中的函数,且在mian函数所在的.obj中声明了,链接器会去找其他.obj文件中找找到那个函数,然后将多个.obj文件中的东西复制到一个exe可执行文件中,我们需要使用其他文件中的函数和变量,这个就是链接器去找的,找到了虽有需要找到然后缝缝补补成一exe

 

链接器会将每个目标文件看成一组外部对象(全局的),每个外部对象看成内存中的一部分,链接器还会处理一些命名冲突

如果一个文件中不允许有两个外部对象名字一样,

 

预处理阶段由预处理器做将源文件作为输入,.i文件作为输出

编译阶段由编译器将,i文件作为输入,.asm文件作为输出

汇编阶段由汇编器将.asm文件作为输入,.obj文件作为输出

以上每个阶段文件生成数,与输出数相同,

链接器是将生成的多个obj文件补合成一个可执行文件

发送在编译期间的错误最容易找,发送链接阶段的错误很让通头痛(如果没有理解链接阶段做了什么)发送了链接错误

那就是噩梦

看我们引用其他文件中的一个外部变量 a,但是我在其他文件中没有定义,然后链接的时候就找到了,然后就给了一个链接错误

当然这是最简单的链接错误

编译器的过程,从源文件到.exe文件经过就几步,是如何进行的

 

如果想要跟加深入了解这写过程是如何做的,建议学习编译原理