gcc/g++是一个编译器,可以将程序文件翻译成可执行文件,gcc用来编译c语言程序,g++用来编译c++程序。
我们了解过,使用语句:gcc test.c ,即会对这个文件进行翻译生成a.out这个可执行文件,如果不想要叫a.out,则可以加上 -o 选项:gcc -o test test.c;注意:-o 选项后面一定要接要生成的文件的名字。
一、程序的翻译
我们知道程序在翻译过程中一共有四个阶段,分别是:
- 预处理(头文件展开,条件编译,宏替换,去注释等)
- 编译(转换为C语言汇编语言)
- 汇编(转换为可重定位目标二进制文件,不可以被执行,windows下生成的是.obj文件)
- 链接(将我们这自己形成的.obj文件和库文件进行合并,形成可执行程序)
1. 预处理(进行宏替换)
- -E :从现在开始进行程序的翻译,预处理做完,就停下来。
预处理生成的文件后缀是 .i
编写一段C语言程序,然后执行语句:
gcc -E test.c -o test.i
可以清楚的看出test.i进行了预处理
2. 编译(生成汇编)
- 在这个阶段,gcc首先要检查代码的规范性、是否有语法错误等,以确定代码的实际要做的工作,在检查无误后,gcc把代码翻译成汇编语言。
- -S :从现在开始进行程序的翻译,当编译做完,就停下来
汇编生成的文件后缀是 .s
执行语句:
gcc -S test.i -o test.s
或
gcc -S test.c -o test.s
可以看出,代码已经被翻译成汇编语言
3. 汇编(生成机器可识别代码)
- 汇编阶段是把编译阶段生成的“-s”文件转成二进制目标文件
- -c :从现在开始进行程序的翻译,当汇编做完,就停下来
汇编生成的文件后缀是 .o
执行语句:
gcc -c test.s -o test.o
或:
gcc -c test.c -o test.o
生成的二进制目标文件:
4. 链接(生成可执行文件或库文件)
- 在成功编译之后,就进入了链接阶段
执行语句:
gcc test.c -o test
可执行文件test内容:
二、函数库
我们之所以能在Linux下进行C、C++代码的编写和编译,是因为Linux系统默认已经携带了语言级别的头文件和语言对应的库。
函数库分为静态库和动态库两种。
- 静态库是指编译链接时,把库文件的代码全部加入到可执行文件中,因此生成的文件比较大,但在运行时也就不再需要库文件了。其后缀名一般是“.a”
- 动态库与之相反,在编译链接是并没有把和库文件的代码加入到可执行文件中,而是程序执行时由运行时链接文件加载库,这样可以节省系统的开销。动态库一般后缀名为“.so”。gcc在编译时默认使用动态库。
- gcc默认生成的二进制程序,是动态链接的,这点可以通过file命令验证。
静态库和静态链接:链接的时候,如果是静态链接,找到静态库,拷贝静态库中的所需要的代码到可执行程序中。
动态库和动态链接:链接的时候,如果是动态链接,找到动态库,拷贝动态库中的所需要的代码的地址到可执行程序中相关的位置。
静态链接成功:程序不依赖任何库,自己就可以独立运行。
动态链接成功:程序还是依赖动态库,一旦动态库缺失,程序就无法运行。
静态库因为自身拷贝的问题,比较浪费空间。动态库因为可以做到被大家共享,所以真正的实现永远都是在库中,程序内部只有地址,比较节省空间,动态库也被叫做共享库。
Linux默认使用的是动态链接和动态库。
使用ldd命令来看一看:
ldd:用来打印或者查看程序运行所需的共享库。
我们如何来对一个.o文件进行静态链接呢?
gcc提供了一个选项:-static,使用它即可进行静态链接,但很多的云服务器默认都只有动态库,没有静态库,所以我们需要安装静态库:
//C语言
sudo yum install -y glibc-static
//C++
sudo yum install -y libstdc++-static
执行语句:
gcc -static test.o -o test-static
生成静态链接的可执行文件
对比动态链接生成的可执行文件的大小和静态链接生成的可执行文件的大小,验证了前面说的静态链接的缺点。
三、gcc选项
- -E 只激活预处理,这个不生成文件,你需要把它重定向到一个输出文件里面
- -S 编译到汇编语言不进行汇编和链接
- -c 编译到目标代码
- -o 文件输出到 文件
- -static 此选项对生成的文件采用静态链接
- -g 生成调试信息。GNU 调试器可利用该信息。
- -shared 此选项将尽量使用动态库,所以生成文件比较小,但是需要系统由动态库.
- -O0
- -O1
- -O2
- -O3 编译器的优化选项的4个级别,-O0表示没有优化,-O1为缺省值,-O3优化级别最高
- -w 不生成任何警告信息。
- -Wall 生成所有警告信息。