编译与链接,咕咕咕

1.翻译环境与运行环境

在ANSIC的任何一种实现中,存在两个不同的环境

(1)翻译环境,在这个环境中源代码被转换为可执行的机器指令(二进制指令)

(2)执行环境,用于实际执行代码

cpp 复制代码
test1.c->编译->链接->可执行程序->输出结果

2.翻译环境

翻译环境由编译和链接两个大的过程组成,编译分解为预处理(预编译),编译,汇编三个过程

cpp 复制代码
test.c->cl.exe(编译器)->test.obj->link.exe(链接器)->xxx.exe(可执行程序)
 ||||||||||||||||||||   @@@@@@@@@@@@@@@@@@@@@
   编译                     链接

一个c语言项目中可能有多个.c文件一起构建。

多个.c文件单独经过编译器,编译处理生成对应的目标文件。(在Windows环境下目标文件后缀为.obj,Linux环境下目标文件的后缀是.o)

多个目标文件和链接库一起经过连接器处理生成最终的可执行程序

链接库是指运行时库(它是支持程序运行的基本函数集合)或第三方库

cpp 复制代码
//gcc为例
源文件,头文件.c->预处理->预处理后中间文件.i->编译->编译后中间文件.s
->汇编->目标文件.o->链接器->可执行程序
                     ^
                     |
                    链接库

预处理阶段,源文件与头文件会变成.i后缀的文件

cpp 复制代码
gcc -E test.c -o test.i

主要处理源文件中#开始的预编译指令:

把所有的#define删除,并展开所有的宏定义。

处理所有的条件编译指令,#if,#ifdef,#else,#endif

处理#include预编译指令,把包含的头文件内容插入到该预编译指令的位置,(递归进行),头文件可能包含头文件

删除所有的注释

添加行号与文件名标识,方便后续编译器生成调试信息等

保留所有的#pragma编译器指令,编译器后续会使用

我们不知道宏定义或头文件是否包含正确时,可以查看预处理后的.i文件

cpp 复制代码
gcc -S test.i -o test.s

编译会对预处理后的文件进行:词法分析,语法分析,语义分析及优化,生成相应的汇编代码文件。

词法分析:把源代码程序输入扫描器,扫描器进行简单的词法分析,把代码中的字符分割成一系列的记号(关键字,标识符,字面量,特殊字符等)

语法分析:语法分析器对扫描产生的记号进行语法分析,产生语法树,这些语法树是以表达式为结点的树

语义分析;由语义分析器来语义分析,即对表达式的语法层面分析。编译器能做的分析是语义的静态分析(包含声明和类型的匹配,类型的转换等),会报告错误的语法信息

汇编,汇编器把汇编代码变成机器可以执行的指令(2进制指令),每一个汇编语句几乎都对应一条机器指令。就是根据汇编指令和机器指令的对照表一一对照翻译,不做指令优化

链接,链接的时候把一堆文件链接在一起才能生成可执行程序。主要包含地址与空间分配,符号决议与重定位等步骤。解决一个项目中多个文件,模块互相调用的问题。

cpp 复制代码
test.c->test.o
#include<stdio.h>
extern int Add(int x,int y);
extern g_val;
int main()
{
int a=10;
int b=20;
int sum=Add(a,b);
printf("%d",sum);
return 0;
}

add.c->add.o
int g_val=1;
int Add(int x,int y)
{
return x+y;
}

在test.c文件中每使用一次Add函数和g_val必粗确切的知道Add和g_val的地址,但是由于每个文件是单独编译的,在编译器编译test.c时并不知道Add和g_val变量的地址,所以暂时把调用Add的指令的目标地址和g_val的地址搁置。等待最后链接时由链接器根据引用的符号Add在其他模块中查找Add的地址,将test.c中所有引用到Add指令重新修正,让他们的目标地址真正为Add函数的地址,这个地址修正的过程叫重定位。(更多的去看《程序员的自我修养》)

3.运行环境

1.程序必须载入内存中,在有操作系统的环境中:一般这个由操作系统完成。在独立的环境中,程序的载入必须由手工安排,也可能是通过可执行代码置入只读内存来完成

2.程序的执行便开始,接着调用main函数

3.开始执行程序代码,程序将使用一个运行时堆栈(stack),存储函数的局部变量和返回地址。程序同时也使用静态内存(static),存储于静态内存中的变量在程序的整个执行过程一直保留他们的值

4.终止程序

相关推荐
Yue丶越1 天前
【C语言】字符函数和字符串函数
c语言·开发语言·算法
蓝牙先生1 天前
简易TCP C/S通信
c语言·tcp/ip·算法
Old_Driver_Lee1 天前
C语言常用语句
c语言·开发语言
松涛和鸣1 天前
从零开始理解 C 语言函数指针与回调机制
linux·c语言·开发语言·嵌入式硬件·排序算法
无限进步_1 天前
C语言动态内存的二维抽象:用malloc实现灵活的多维数组
c语言·开发语言·数据结构·git·算法·github·visual studio
星轨初途1 天前
数据结构排序算法详解(5)——非比较函数:计数排序(鸽巢原理)及排序算法复杂度和稳定性分析
c语言·开发语言·数据结构·经验分享·笔记·算法·排序算法
f***01931 天前
CC++链接数据库(MySQL)超级详细指南
c语言·数据库·c++
小曹要微笑1 天前
STM32F7 时钟树简讲(快速入门)
c语言·stm32·单片机·嵌入式硬件·算法
前端世界1 天前
float 还是 double?用储罐体积计算带你看懂 C 语言浮点数的真实世界坑
java·c语言·开发语言
小青龙emmm1 天前
2025级C语言第二次周测(国教专用)题解
c语言·开发语言·算法