C/C++程序编译、链接和装入

C++程序编译过程(以g++为例)

预处理
处理#开始的命令,得到不包含#指令的.i文件,包括替换宏、引入头文件、去除注释等;使用预处理器cpp。
编译
分析.i文件的语法和词法,确定是否所有指令都符合规则,随后翻译成汇编代码,得到.s文件;使用编译器egcs。
汇编
将汇编代码转换为目标代码(机器语言二进制代码),生成.o文件;使用汇编器as。
链接
将外部库函数代码添加到可执行文件中;使用链接器ld。

C语言程序编译过程

链接(Link)

一个完整的程序往往被分割为若干个独立的部分并行开发,而各个模块间通过函数接口或全局变量进行通讯。编译器只能在一个模块内部完成符号名到地址的转换工作,不同模块之间的符号解析需要由链接器来完成。
链接器的主要工作包括符号解析和重定位:
符号解析
到别的模块中查找本模块中未定义过的函数或者全局变量。
重定位
编译器在编译生成目标文件时,通常都使用从零开始的相对地址。然而,在链接过程中,链接器将从一个指定的地址开始,根据输入的目标文件的顺序以段为单位将它们一个接一个的拼装起来。除了目标文件的拼装之外,在重定位的过程中还完成了两个任务:一是生成最终的符号表;二是对代码段中的某些位置进行修改,所有需要修改的位置都由编译器生成的重定位表指出。

目标文件

目标文件从结构上讲是可执行文件格式(在linux下是ELF格式),只是还没有经过链接处理,有些符号和地址还没有经过修正。目标文件有未解决符号表(unresolved symbol table)、导出符号表(export symbol table)和地址重定向表(address redirect table)。其中 :
未解决符号表:列出了本单元里有引用但是不在本单元定义的符号及其出现的地址。
导出符号表:提供了本编译单元具有定义,并且可以提供给其他编译单元使用的符号及其在本单元中的地址。
地址重定向表:提供了本编译单元所有对自身地址的引用记录。

链接器的工作顺序

当链接器进行链接的时候,首先决定各个目标文件在最终可执行文件里的位置。然后访问所有目标文件的地址重定向表,对其中记录的地址进行重定向(加上一个偏移量,即该编译单元在可执行文件上的起始地址)。然后遍历所有目标文件的未解决符号表,并且在所有的导出符号表里查找匹配的符号,并在未解决符号表中所记录的位置上填写实现地址。最后把所有的目标文件的内容写在各自的位置上,就生成一个可执行文件。

静态库

静态库在编译过程中会被链接进目标可执行文件(部分还是全部?),在可执行文件运行的过程中,不需要同时携带静态库。

  • 命名格式
    lib+库的名字+.a
  • 发布时需要携带头文件
  • 制作
    第一步:得到.o文件
gcc <源文件> -c -I <path/to/include>

第二步:创建静态库

ar rcs libMyLib.a <目标文件.o>
  • 使用
    第一种方法:
gcc <源文件> -L<静态库路径> -l<静态库名> -I <头文件目录> -o <可执行文件名>

第二种方法:

gcc <源文件> -I <头文件目录> libxxx.a -o <可执行文件名>

动态库

动态库在编译过程中不会被链接进可执行文件。能够节省空间并且更加灵活(比如可以独立于可执行文件更新,存疑)。

  • 命名格式
    lib+库的名字+.so
  • 制作

第一步:生成与位置无关的.o文件

gcc -fPIC <源文件.c> -I <头文件目录> -c

第二步:创建动态库

gcc -shared -o libMyTest.so <目标文件.o>
  • 使用

第一种方法:

gcc <源文件> -L<动态库路径> -l<动态库名> -I <头文件目录> -o <可执行文件名>

第二种方法:

gcc <源文件> -I <头文件目录> libxxx.so -o <可执行文件名>
  • 查找方式

编译时(以ld链接器为例):

  1. -L选项声明的路径
  2. /usr/lib /usr/local/lib

运行时(以Debian为例):

  1. 库的DT_RPATH属性
  2. 可执行文件的DT_RPATH属性
  3. LD_LIBRARY_PATH环境变量
  4. 可执行文件的DT_RUNPATH属性
  5. /etc/ld.so.cache
  6. /lib和/usr/lib

装入(Load)

多道程序环境下,程序是并发执行的,要使程序运行,必须为之创建进程,而创建进程第一件事就是将程序和数据装入内存。

链接和装入的不同阶段

静态链接、静态装入
将所有的目标文件连接成一个可执行映像,随后在创建进程时将该可执行影像一次性全部装入内存。这样导致硬盘和内存利用效率都不高。
静态链接、动态装入
一个函数只有在被调用时,其所在的模块才会被装入内存(同时更新此程序的地址表)。这样提高了内存利用效率。
动态链接、动态装入
在使用动态链接时,需要在程序影像中每个调用库函数的地方打一个桩(stub)。桩是一小段代码,用于定位已装入内存的相应的库;如果所需的库还不在内存中,stub将指出如何将该函数所在的库装入内存。此外,用到同一个库的进程将通过地址映射到相同的库的实现。动态链接的这一特性对于库的升级(比如错误的修正)是至关重要的。当一个库升级到一个新版本时,所有用到这个库的程序将自动使用新的版本。如果不使用动态链接技术,那么所有这些程序都需要被重新链接才能得以访问新版的库。为了避免程序意外使用到一些不兼容的新版的库,通常在程序和库中都包含各自的版本信息。内存中可能会同时存在着一个库的几个版本,但是每个程序可以通过版本信息来决定它到底应该使用哪一个。如果对库只做了微小的改动,库的版本号将保持不变;如果改动较大,则相应递增版本号。因此,如果新版库中含有与早期不兼容的改动,只有那些使用新版库进行编译的程序才会受到影响,而在新版库安装之前进行过链接的程序将继续使用以前的库。这样的系统被称作共享库系统。

gcc常用选项

-o 指定目标名称
-c 只激活预处理、编译和汇编,得到目标文件
-S 只激活预处理和编译,得到汇编代码
-E 只激活预处理,得到.i文件
-C 在预处理的时候,不删除注释信息
-static 禁止使用动态库
-share 尽量使用动态库
-l 指定头文件目录
-D 编译时定义宏
-g 包含调试信息

参考文献
https://www.ibm.com/developerworks/cn/linux/l-dynlink/index.html
https://blog.csdn.net/edisonlg/article/details/7081357
https://blog.csdn.net/HappyToEat/article/details/53163385

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,125评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,293评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,054评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,077评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,096评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,062评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,988评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,817评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,266评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,486评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,646评论 1 347
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,375评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,974评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,621评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,796评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,642评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,538评论 2 352