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

2024-03-17 09:58

本文主要是介绍编译器的过程,从源文件到.exe文件经过就几步,是如何进行的,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

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

1.预处理

2.编译

3.汇编

4.链接

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

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

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

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

 设置好后点重新生成,可以看到给两个源文件都生成了.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);
 

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

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

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

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

 

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

 

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

 

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

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

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

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

可能有的人有些奇怪汇编文件不是.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文件经过就几步,是如何进行的的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/818609

相关文章

Redis中Hash从使用过程到原理说明

《Redis中Hash从使用过程到原理说明》RedisHash结构用于存储字段-值对,适合对象数据,支持HSET、HGET等命令,采用ziplist或hashtable编码,通过渐进式rehash优化... 目录一、开篇:Hash就像超市的货架二、Hash的基本使用1. 常用命令示例2. Java操作示例三

Redis中Set结构使用过程与原理说明

《Redis中Set结构使用过程与原理说明》本文解析了RedisSet数据结构,涵盖其基本操作(如添加、查找)、集合运算(交并差)、底层实现(intset与hashtable自动切换机制)、典型应用场... 目录开篇:从购物车到Redis Set一、Redis Set的基本操作1.1 编程常用命令1.2 集

Linux下利用select实现串口数据读取过程

《Linux下利用select实现串口数据读取过程》文章介绍Linux中使用select、poll或epoll实现串口数据读取,通过I/O多路复用机制在数据到达时触发读取,避免持续轮询,示例代码展示设... 目录示例代码(使用select实现)代码解释总结在 linux 系统里,我们可以借助 select、

k8s中实现mysql主备过程详解

《k8s中实现mysql主备过程详解》文章讲解了在K8s中使用StatefulSet部署MySQL主备架构,包含NFS安装、storageClass配置、MySQL部署及同步检查步骤,确保主备数据一致... 目录一、k8s中实现mysql主备1.1 环境信息1.2 部署nfs-provisioner1.2.

Python打包成exe常用的四种方法小结

《Python打包成exe常用的四种方法小结》本文主要介绍了Python打包成exe常用的四种方法,包括PyInstaller、cx_Freeze、Py2exe、Nuitka,文中通过示例代码介绍的非... 目录一.PyInstaller11.安装:2. PyInstaller常用参数下面是pyinstal

MyBatis/MyBatis-Plus同事务循环调用存储过程获取主键重复问题分析及解决

《MyBatis/MyBatis-Plus同事务循环调用存储过程获取主键重复问题分析及解决》MyBatis默认开启一级缓存,同一事务中循环调用查询方法时会重复使用缓存数据,导致获取的序列主键值均为1,... 目录问题原因解决办法如果是存储过程总结问题myBATis有如下代码获取序列作为主键IdMappe

Python进行word模板内容替换的实现示例

《Python进行word模板内容替换的实现示例》本文介绍了使用Python自动化处理Word模板文档的常用方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友... 目录技术背景与需求场景核心工具库介绍1.获取你的word模板内容2.正常文本内容的替换3.表格内容的

linux部署NFS和autofs自动挂载实现过程

《linux部署NFS和autofs自动挂载实现过程》文章介绍了NFS(网络文件系统)和Autofs的原理与配置,NFS通过RPC实现跨系统文件共享,需配置/etc/exports和nfs.conf,... 目录(一)NFS1. 什么是NFS2.NFS守护进程3.RPC服务4. 原理5. 部署5.1安装NF

Git进行版本控制的实战指南

《Git进行版本控制的实战指南》Git是一种分布式版本控制系统,广泛应用于软件开发中,它可以记录和管理项目的历史修改,并支持多人协作开发,通过Git,开发者可以轻松地跟踪代码变更、合并分支、回退版本等... 目录一、Git核心概念解析二、环境搭建与配置1. 安装Git(Windows示例)2. 基础配置(必

MySQL使用EXISTS检查记录是否存在的详细过程

《MySQL使用EXISTS检查记录是否存在的详细过程》EXISTS是SQL中用于检查子查询是否返回至少一条记录的运算符,它通常用于测试是否存在满足特定条件的记录,从而在主查询中进行相应操作,本文给大... 目录基本语法示例数据库和表结构1. 使用 EXISTS 在 SELECT 语句中2. 使用 EXIS