当前位置: 首页 > news >正文

【C语言】C语言 4 个编译过程详解

C语言的编译过程涉及几个关键步骤、概念和细节,每个步骤都有助于将人类可读的源代码转换为可执行的机器码。以下是详细的解释和示例:

一、什么是编译?

编译是将源代码转换为目标代码的过程。它是在编译器的帮助下完成的。编译器检查源代码是否存在语法或结构错误,如果源代码没有错误,则生成目标代码。

在这里插入图片描述

c 编译过程将作为输入的源代码转换为目标代码或机器代码。编译过程可分为预处理、编译、汇编和链接四个步骤。
在这里插入图片描述

二、编译过程中的步骤:

  1. 预处理

    • 概念:预处理器 (cpp) 处理以 # 开头的指令,如 #include#define#ifdef。它通过展开宏和包含头文件来准备源代码进行编译。
    • 细节:将源代码转换为预处理后的中间形式。源代码是在文本编辑器中编写的代码,源代码文件的扩展名为 .c。此源代码首先传递给预处理器,然后预处理器扩展此代码。展开代码后,扩展的代码将传递给编译器。
    • 示例
      #include <stdio.h>
      #define MAX 100int main() {printf("Max value: %d\n", MAX);return 0;
      }
      
    • 后缀.i (生成的预处理后的源文件)。
  2. 编译

    • 概念:编译器 (gcc, clang) 将预处理后的源代码翻译成特定于目标体系结构的汇编语言。
    • 细节:执行词法分析、语法分析、语义分析和优化。由预处理器扩展的代码将传递给编译器。编译器将此代码转换为汇编代码。或者我们可以说 C 编译器将预处理的代码转换为汇编代码。
    • 示例
      mov eax, 42
      
    • 后缀.s (生成的汇编代码文件)。
  3. 汇编

    • 概念:汇编器 (as) 将汇编代码转换为可重定位的机器码(目标代码)。
    • 细节:将汇编指令转换为机器码指令。使用汇编程序将汇编代码转换为目标代码。汇编程序生成的目标文件的名称与源文件的名称相同。在 DOS 中,目标文件的扩展名是 .obj,在 UNIX 中,扩展名是 .o。如果源文件的名称为 hello.c,则目标文件的名称将为 hello.obj
    • 示例
      00000000: B8 2A 00 00 00        mov eax, 0x2a
      
    • 后缀.o (生成的目标文件)。
  4. 链接

    • 概念:链接器 (ld) 将多个目标文件组合在一起,并解决外部引用,生成最终的可执行代码。
    • 细节:执行符号解析、为变量和函数分配最终地址,并链接必要的库文件。主要是,所有用 C 编写的程序都使用库函数。这些库函数是预先编译的,并且这些库文件的目标代码以 .lib(或 .a)扩展名存储。链接器的主要工作是将库文件的目标代码与我们程序的目标代码相结合。有时,当我们的程序引用其他文件中定义的函数时,会出现这种情况;那么 linker 在这方面起着非常重要的作用。它将这些文件的目标代码链接到我们的程序。因此,我们得出结论,链接器的工作是将程序的目标代码与库文件和其他文件的目标代码链接起来。链接器的输出是可执行文件。可执行文件的名称与源文件的名称相同,但仅在扩展名上有所不同。在DOS中,可执行文件的扩展名为 .exe,而在UNIX中,可执行文件可以命名为 a.out。例如,如果我们在程序中使用 printf() 函数,则链接器会将其关联的代码添加到输出文件中。
    • 示例
      gcc main.o helper.o -o myprogram
      
    • 后缀:(可执行文件通常没有特定的后缀,常见的是没有后缀或 .out)。

三、其他概念和细节:

  • 编译器优化:提升代码性能和大小。
  • 目标文件:包含机器码和符号。
  • 静态与动态链接:静态链接库和动态链接库的嵌入与运行时链接。
  • 可执行文件格式:各种操作系统特定的格式(如 ELF、PE)。
  • .hex.bin 文件通常不是C语言编译过程的直接产物,而是在嵌入式系统开发中常见的文件格式,用于存储程序或数据的二进制表示。这些文件通常是在程序已经编译、链接并生成了可执行文件之后,通过特定的工具或者转换过程生成的。因此,它们不属于C语言编译过程的标准阶段,但是在嵌入式开发中是非常常见的文件格式。
  • .hex 文件:通常用于存储以十六进制格式表示的固件或程序映像,适用于多种嵌入式系统和芯片编程器。
  • .bin 文件:通常用于存储原始的二进制数据,可以包括程序或者数据文件,常见于嵌入式系统和低级系统编程中。

示例 1 解释:

对于给定的C程序示例:

#include <stdio.h>
#define MAX 100int main() {printf("Max value: %d\n", MAX);return 0;
}
  • 预处理:展开 #include <stdio.h> 并将 MAX 替换为 100
  • 编译:将C源代码转换为汇编语言。
  • 汇编:将汇编指令转换为机器码。
  • 链接:将机器码与标准库函数(如 printf)结合,生成最终的可执行文件。

示例 2 解释:

hello.c :

#include <stdio.h>  
int main()  
{  printf("Hello World !");  return 0;  
} 

现在,我们将创建上述程序的流程图:

在这里插入图片描述

在上面的流程图中,执行程序需要执行以下步骤:

  1. 首先,将输入文件(即 hello.c)传递给预处理器,预处理器将源代码转换为扩展的源代码。扩展源代码的扩展将是 hello.i
  2. 扩展的源代码将传递给编译器,编译器将此扩展的源代码转换为汇编代码。程序集代码的扩展名为 hello.s
  3. 然后,将此汇编代码发送到汇编程序,汇编程序将程序集代码转换为目标代码。
  4. 创建目标代码后,链接器将创建可执行文件。然后,加载程序将加载执行的可执行文件。

理解这些步骤有助于优化代码、调试问题,并了解C编程中的平台特定细节。

相关文章:

  • 一站式采购!麒麟信安CentOS安全加固套件上架华为云云商店
  • 简易电阻、电容和电感测量仪-FPGA
  • [leetcode]max-consecutive-ones 最大连续1的个数
  • 【高级篇】第10章 Elasticsearch 集群管理与扩展
  • 数据结构(Java):迭代器遍历【底层源码解析】
  • 2024年高校辅导员考试题库及答案
  • c#上位机与PLC通讯心跳的实现方法
  • Go内存回收
  • Python获取QQ音乐歌单歌曲
  • 数据特征采样在 MySQL 同步一致性校验中的实践
  • 计算机未来大方向的选择
  • MViT(ICCV 2021, Meta)论文解读
  • 阶段三:项目开发---大数据系统基础环境准备:任务1:准备系统运行的先决条件
  • vue项目创建+eslint+Prettier+git提交规范(commitizen+hooks+husk)
  • React
  • angular组件开发
  • canvas 五子棋游戏
  • JavaScript 基础知识 - 入门篇(一)
  • Java-详解HashMap
  • learning koa2.x
  • Mocha测试初探
  • niucms就是以城市为分割单位,在上面 小区/乡村/同城论坛+58+团购
  • socket.io+express实现聊天室的思考(三)
  • TypeScript实现数据结构(一)栈,队列,链表
  • 官方新出的 Kotlin 扩展库 KTX,到底帮你干了什么?
  • 基于web的全景—— Pannellum小试
  • 简单实现一个textarea自适应高度
  • 经典排序算法及其 Java 实现
  • 数据仓库的几种建模方法
  • 推荐一个React的管理后台框架
  • 小试R空间处理新库sf
  • 云大使推广中的常见热门问题
  • 最近的计划
  • HanLP分词命名实体提取详解
  • 小白应该如何快速入门阿里云服务器,新手使用ECS的方法 ...
  • ​草莓熊python turtle绘图代码(玫瑰花版)附源代码
  • #pragma once
  • #每天一道面试题# 什么是MySQL的回表查询
  • $.extend({},旧的,新的);合并对象,后面的覆盖前面的
  • (3) cmake编译多个cpp文件
  • (顶刊)一个基于分类代理模型的超多目标优化算法
  • (二)【Jmeter】专栏实战项目靶场drupal部署
  • (附源码)计算机毕业设计SSM基于健身房管理系统
  • (规划)24届春招和25届暑假实习路线准备规划
  • (算法二)滑动窗口
  • (五)关系数据库标准语言SQL
  • (转)GCC在C语言中内嵌汇编 asm __volatile__
  • (转)scrum常见工具列表
  • (转)如何上传第三方jar包至Maven私服让maven项目可以使用第三方jar包
  • .net core 调用c dll_用C++生成一个简单的DLL文件VS2008
  • .NET Windows:删除文件夹后立即判断,有可能依然存在
  • .Net多线程Threading相关详解
  • .NET连接数据库方式
  • // an array of int
  • @DateTimeFormat 和 @JsonFormat 注解详解