文章目录
gcc如何完成
格式: gcc [选项] [要编译的文件] [选项] [目标文件]
(1)预处理(进行宏替换)
- 预处理功能主要包括宏定义,文件包含,条件编译,去注释等。
- 预处理指令是以#号开头的代码行。
- 实例:
gcc –E hello.c –o hello.i
- 选项
-E
,该选项的作用是让 gcc 在预处理结束后停止编译过程。 - 选项
-o
是指目标文件,.i
文件为已经过预处理的C原始程序
(2)编译(生成汇编)
- 在这个阶段中,gcc 首先要检查代码的规范性、是否有语法错误等,以确定代码的实际要做的工作,在检查无误后,gcc 把代码翻译成汇编语言。
- 用户可以使用
-S
选项来进行查看,该选项只进行编译而不进行汇编,生成汇编代码。 - 实例:
gcc –S hello.i –o hello.s
(3)汇编(生成机器可识别代码)
- 汇编阶段是把编译阶段生成的
.s
文件转成目标文件 - 读者在此可使用选项
-c
就可看到汇编代码已转化为.o
的二进制目标代码了 - 实例:
gcc –c hello.s –o hello.o
由于是二进制文件,一般的打开方式会所以会乱码。
使用命令od hello.o | less
即可查阅二进制文件。
(4)链接(生成可执行文件或库文件)
- 在成功编译之后,就进入了链接阶段。
- 实例:
gcc hello.o –o hello
重要概念:函数库
gcc如何链接函数
- 我们的C程序中,并没有定义
printf
的函数实现,且在预编译中包含的stdio.h
中也只有该函数的声明,而没有定义函数的实现,那么,是在哪里实现printf
函数的呢? - 最后的答案是:系统把这些函数实现都被做到名为 libc.so.6 的库文件中去了,在没有特别指定时,gcc 会到系统默认的搜索路径 /usr/lib 下进行查找,也就是链接到 libc.so.6 库函数中去,这样就能实现函数printf了,而这也就是链接的作用
静态库与动态库
-
静态库是指编译链接时,把库文件的代码全部加入到可执行文件中,因此生成的文件比较大,但在运行时也就不再需要库文件了。其后缀名为
.a
。输入gcc hello.c -o hello -static
,采用静态链接 -
动态库在编译链接时并没有把库文件的代码加入到可执行文件中,而是在程序运行时由运行时链接文件加载库,这样做可以节省系统开销。动态库后缀名一般为
.so,gcc
在编译时默认使用动态库,完成链接之后,就生成了可执行文件。
动态链接形成的程序体积较小,比较节省资源,但是一旦库丢失,程序就不可以运行了;而静态形成的程序的体积很大,但具有独立性,即便库丢失,也不影响程序运行。
file指令与ldd指令
file
是一个查看文件类型的命令,如下:
我们可以看到这个文件是一个elf形式的可执行程序,64位,平台为x86,与本博客契合的是它是一个动态链接,那么所对应的就是利用了动态库。
ldd
是可以查看一个可执行程序所依赖的库是静态库还是动态库。
主要关注这一行,即libc.so.6 => /lib64/libc.so.6
一般来说,我们库的前缀lib以及后缀.so.6(.a)删除这些,余下的即为语言类型,很显然,这里所用的是c语言的库函数。