日期:2023-02-11 19:04:13 来源:腾讯云
本文利用C语言介绍源文件是如何一步一步地编译、链接成为可执行程序的。
(资料图)
编译链接本身分为许多个过程,但从整体上可以概括为以下三点:
程序的每个源文件分别
通过编译器的编译,生成对应的目标文件。每个目标文件通过链接器链接在一起,形成一个独立单一而完整
的可执行文件。链接时会在链接库内搜索C标准库及程序员个人函数库
中的函数,并将其链接至可执行程序。整体过程如下图所示:
预处理,又叫预编译,源文件在进入编译器后要首先进行预处理操作,告诉编译器如何预处理的代码被称为预处理指令,常见的有头文件的包含,宏定义等。
经过预处理后的文件后缀会变为i
,文件以i
为后缀代表其已经过预处理,但未进行下一阶段的编译。
在我们写代码时,包含头文件的是必不可少的,而在预处理中:包含头文件的操作等同于拷贝头文件中的代码,随后将源文件中的代码和头文件中的代码一起放入后缀为i
的文件中。
预处理前:
预处理后:
相当于将头文件中的代码拷贝至源文件的头部,同时删掉#include"head.h"
其效果与下面的代码相同,此时的文件后缀为i
。
struct MyStruct{int a;int b;};int main(){struct MyStruct s;return 0;}
在预处理时,源文件中宏定义内容会在i
文件中被直接替换。
预处理前:
#include#define MAX 100int main(){int a = MAX;printf("%d\n", a);return 0;}
预处理后:
预处理后MAX
会被直接替换为100
,同时删除前面的#define MAX 100
,其效果与下图代码相同。
在预处理时,编译器还会删除//
后面的内容以及被/* */
包括起来的内容,以便于接下来的编译阶段。
如图:
可以看到,不管是头文件、宏定义、还是注释,都是编译器对文本
的操作。
因此可以说:预处理阶段是编译器对文本的操作阶段。
当源文件经过预处理后,便开始第二个阶段——编译,经过编译的文件的后缀将变成s
,文件后缀为s
代表其已经经过第二阶段的编译,但未进行汇编操作。
编译阶段会将C语言代码进行语法分析
、词法分析
、词义分析
,以及符号汇总
。
编译阶段从整体上看是对C语言代码进行分析解读
由于该阶段涉及编译的原理,因此只了解大概即可,详见书籍《编译原理》。
当源文件经过第一、二阶段后,便开始第三个阶段——汇编,经过汇编的文件的后缀将变成o或obj
,即开头所说的目标文件
,文件后缀为o或obj
代表其至少已经经过三个阶段的编译。
PS:在VS环境下生成的目标文件后缀为obj
,在GCC环境下生成的目标文件后缀为o
。
在第二阶段的编译中会将代码中的全局变量
,函数名
等汇总,并在汇编阶段形成类似一张表格。
收集以下代码中的全局变量与函数名:
形成表格:
符号表的详细作用在链接时详细介绍。
名 | 址 | ... |
---|---|---|
val | 0X0012FF40 | |
add | 0X0012FF80 | |
main | 0X0012FF20 |
链接主要包括合并段表
、符号表的合并及重定位
两大过程。
下面依次介绍。
在源文件经过编译过程后文件后缀变为o
或obj
后,其内部的代码已经变成二进制。
此时,文件会将内部的二进制代码按类型
划分成许多部分,每一部分就被称为一段。
如图:
前文我们说到,多个目标文件经过链接器后整合为一个独立而完整的文件,因此链接中合并段表的意思就是把多个目标文件中对应的段,合并在一个目标文件内。
顾名思义,将每个目标文件中的符号表合并成一个目标文件的符号表。
如果在写代码时发生函数定、声明、使用等不在同一源文件等情况,那么多个目标文件的符号表中可能都会有相同的函数名及无效的地址,像这样:
如上图,在两个文件中都有add的存在,那么,在head.obj
和源.obj
的符号表中都会有收录add。
但因源.c
中没有对add进行实现,因此在汇总源.c
的符号表时,其地址会被填入一个无效的地址。
在链接过程中,链接器不仅会对符号表合并,也会将多余的符号删除,同时将每个符号的地址都定位为所有地址中有效的那一个。
如果在写代码时不小心将函数名写错了,像这样:
那么由于在head.h
汇总符号表时找不到代码实现,就会把Add
的地址赋为一个无效的地址,像这样:
名 | 址 |
---|---|
Add | 0X00000000 |
add | 0X0012FF40 |
但是,在符号表的重定位时,也没有重定位Add的有效地址,那么链接就无法完成,就会报出典型的错误:
PS:函数声明extern
的作用就是告诉编译器,遇到无法解释的符号先别急着报错,后面会有定义的。
多个源文件经过预处理(头文件包含、宏定义替换、注释的删除),编译(语法分析、词法分析、词义分析、符号汇总),汇编(二进制指令翻译、形成符号表)后成为一个目标文件。
多个目标文件在链接库的帮助下完成合并段表,符号表合并,符号表重定位后整合为一个目标文件。
到最后变成一个可执行的exe文件。
感谢您的阅读与耐心~
【环球热闻】详解源文件编译链接至可执行程序的每一步
世界快消息!一年四季有味道吗?岚山的四季之味都在这里了
【世界速看料】为什么女生会送男生花
天天要闻:刘复国:侯友宜棋子说反映民意 有战略眼光
世界百事通!中国队对韩国队世预赛直播_中国队对韩国队
世界要闻:写作业送什么礼物
【全球独家】国务委员是什么级别的干部
天天实时:湿气重的症状_湿气重怎么调理
世界资讯:弹出拨号连接怎么取消_弹出拨号连接
当前速递!乔尔diu_乔尔马登
今日播报!lovey_dovey
环球消息!海外房产品牌百度推广如何将故事与销融为一体
【天天速看料】猪大肠怎么卤制 猪大肠卤多长时间最好吃
当前看点!新泉股份(603179)2月10日主力资金净买入736.64万元
【世界时快讯】mt探伤检测是什么意思_噪声检测lex 8h是什么意思
环球快消息!中年衣服哪个牌子好 中年衣服十大品牌排行榜
【环球快播报】新生儿黄疸眼睛会变黄吗如何预防呢
天天新资讯:全国女足锦标赛分组及赛程公布:U20女青位居E组,首阶段即将开战
观察:怎么和QQ好友一起看动画片
全球今日报丨刘亚仁中国代言被撤!这条视频一同冲上热搜……
信息:欧阳娜娜王俊凯_欧阳娜娜王俊凯
环球今亮点!数码相机什么牌子好用性价比高_数码相机什么牌子好
环球消息!队报:梅西腿筋受伤,欧冠出战拜仁存疑
世界热文:“象口”之地 暹罗湾
今日关注:新亚_关于新亚的介绍
天天看点:茄子用英语怎么说_茄子英语读音介绍
每日头条!中药里的“解郁高手”,胸闷、胸胀、心抑郁都能用,代茶饮,有效
世界头条:西部乱不乱篮网说了算,杜兰特加盟太阳,篮网用杜欧完成2换12
【独家】甲丁(豫)
世界速递!夏邑县:党建引领产业高质量发展