Linux编译器——gcc/g++
程序编译背景知识
程序的编译分为四步:
预处理:主要完成的是,头文件展开,宏替换,注释删除,条件编译,预处理完的文件后缀是.i
编译:把C语言翻译成汇编语言,翻译完成后生成的文件后缀是.s
汇编:把汇编翻译成二进制目标文件,翻译完的文件就是可重定向二进制目标文件,后缀为.o
链接:将生成的二进制目标文件和C语言标准库里面的代码进行结合最后生成可执程序
gcc指令
预处理:
指令:gcc -E test.c -o test.i
**-E **编译器做完预处理的就停下来
**-o **指明生成的临时文件的名称(test.i)
预处理完之后文件里面还是C语言。
编译:
指令:gcc -S test.i -o test.s
**-S **编译器做完编译工作就停下来
编译过程将C语言翻译成了汇编语言
汇编:
指令:gcc -c test.c -o test.o
**-c **编译器做完汇编过程就停下来
test.o里面保存的就是二进制码,用vim打开时是乱码,od test.o 可以按二进制方式查看文件。
链接:
指令:gcc test.o -o mytest
mytest就是可执行二进制文件(库+自己的代码生成的)
链接其实就是将库中的代码和自己的代码合并的过程。
指令演示:
gcc -E test.c -o test.i
预处理后的文件,从原来的几十行变成了八百多行,多出来的内容都是头文件stdio里面展开的内容,宏被直接替换到代码里面,条件编译也完成了。并且预处理后的文件还是C语言的。
gcc -S test.i -o test.s
编译后文件里的内容变成了汇编语言,movl和call等等这些都是助记符。
gcc -c test.c -o test.o
test.o是二进制目标文件,所以打开就是乱码
od test.o
gcc test.o -o mytest
关于宏的问题,在Linux我们在编译的时候可以指定宏定义。
将代码中的宏定义删掉。
gcc可以直接一步就完成预处理编译汇编的过程,后面的选项顺序可以颠倒,只要-o后面一定是目标可执行程序的名称即可。
此时没有宏定义MAX所以打印的就是这句话了。
动态链接和静态链接
链接的本质,就是调用库函数的时,代码和标准库是如何关联起来的。
链接这里存在动态链接和静态链接,同时也存在动态库和静态库。
动态链接的时候是在程序执行到库函数调用位置的时候跳转到库里面执行库的代码。这个过程称为动态跳转。
由此也引出一个问题,程序是如何知到库在哪?
- 编译器中有一个链接器,使用链接器就可以和库建立链接,调用库函数的时候直接跳转即可。
优点:形成的可执行程序小,可以节省资源(内存,硬盘,网络资源等)
静态链接是将调用的库函数代码拷贝到你的代码中。
所以静态链接的时候不是产生关联,而是直接将调用到的库函数拷贝到程序中完成静态链接。
优点:不受库升级或者删除的影响
缺点:
- 形成的可执行程序体积太大,会占用大量的内存,硬盘,网络资源。
- 出席那了大量重复的代码(不同的文件中调用了同一个函数,该函数被拷贝多次,代码膨胀)
通过file指令可以查看可执行程序的详细信息
file [可执行程序名]
这个词组翻译后就是动态链接的意思,前面的信息就是可执行程序的体系架构。
ldd [可执行程序名]
通过ldd指令查看可执行程序所依赖的动态库列表(必须是动态链接的可执行程序,静态的查看不到)
在这三个库中需要着重关注这个库。后面的地址就是调用的方法在那个位置,方便执行的时候进行跳转。
Linux下库的命名:
动态库:libXXXXX.so (以lib开头和.so结尾的库就是动态库,去掉lib和.so的后缀,剩下的就是库的名称。
所以上面圈出来的动态库的名称就是c,也就是c标准库。
静态库:libXXXXX.a (同样去掉前缀lib和后缀.a剩下的就是静态库的名称。)
下面来看一下动态链接生成的可执行程序和静态链接形成的可执行程序有什么区别。
使用指令:gcc test.c -o mytest-s -static就可指定编译器使用静态链接方式。
很明显静态链接生成的可执行程序的大小要比动态链接的大了100倍左右。
静态链接的文件不可以使用ldd
爆出,该文件不是一个可执行程序的错误。
静态链接的时候如果系统没有静态库会报错
找不到这个静态库。
sudo yum install -y glibc-static下载C语言静态库。
sudo yum install -y libstdc+±static 下载C++的静态库
Linux系统中一般自带C语言的动态库,因为有很多指令都是用C语言写的,所以如果删除了C语言动态库很可能导致大多数程序都无法使用了。
C语言的动态库虽然被很多程序都使用但是动态库只有一个,所有的程序都是共用c标准库的。所以动态库又被称为共享库。
windows系统下
动态库的后缀是.dll
静态库的后最是.lib
系统为了支持编程给用户提供了:
- 标准库的.h文件(头文件里面是声明,告诉用户如何使用)
- 标准的动静态库(库函数的实现)
在使用gcc进行链接的时候如果只有一个动态库或者一个静态库,那么就用这个唯一的库进行链接,如果两个库都有,那么默认使用动态库。
Linux下运行其他语言
Linux并不是只能运行C语言或者C++,其他的语言比如java,python,php等等都是可以的。
使用python解释器直接就可以运行,因为py是脚本语言,所以不需要编译,甚至只要给test.py这个文件加上可执行权限就可以直接运行。
还有shell脚本和php等等脚本语言都是可以直接运行的。