gcc的编译过程分为四个过程,即预处理(也称预编译,Preprocessing)、编译(Compilation)、汇编 (Assembly)和连接(Linking)。
例如,编译文件test.c,那么整个过程如下:
1.预处理
gcc -E test.c -o test.i 或 gcc -E test.c
可以输出test.i文件中存放着test.c经预处理之后的代码。打开test.i文件,看一看,就明白了。后面那条指令,是直接在命令行窗口中输出预处理后的代码.
gcc的-E选项,可以让编译器在预处理后停止,并输出预处理结果。在本例中,预处理结果就是将stdio.h 文件中的内容插入到test.c中了。
2.编译
预处理之后,可直接对生成的test.i文件编译,生成汇编代码:
gcc -S test.i -o test.s
gcc的-S选项,表示在程序编译期间,在生成汇编代码后,停止,-o输出汇编代码文件。
3.汇编
对于上一步骤生成的汇编代码文件test.s,gas汇编器负责将其编译为目标文件,如下:
gcc -c test.s -o test.o
4.链接
gcc连接器是gas提供的,负责将程序的目标文件与所需的所有附加的目标文件连接起来,最终生成可执行文件。附加的目标文件包括静态连接库和动态连接库。
对于上一步骤生成的test.o,将其与C标准输入输出库进行连接,最终生成程序test。
gcc test.o -o test
最后在命令行窗口中,执行./test,即可得到结果。
若进行多个程序文件编译
通常整个程序是由多个源文件组成的,相应地也就形成了多个编译单元,使用GCC能够很好地管理这些编译单元。假设有一个由test1.c和 test2.c两个源文件组成的程序,为了对它们进行编译,并最终生
成可执行程序test,可以使用下面这条命令:gcc test1.c test2.c -o test
如果同时处理的文件不止一个,GCC仍然会按照预处理、编译和链接的过程依次进行。如果深究起来,上面这条命令大致相当于依次执行如下三条命令:
gcc -c test1.c -o test1.o
gcc -c test2.c -o test2.o
gcc test1.o test2.o -o test
gcc -D选项
gcc的-D选项可以定义宏
C语言源程序中有#define定义的宏,可以起到替换、条件编译的功能;定义宏的方式是放在头文件或者C文件中。gcc提供了另外一种宏定义的方法,当然可以想象这样的宏定义方法并不是很好用,但毕竟编译器给我们提供了额外的宏定义的方法。假设程序需要很多宏,不可能这些宏都在编译器中定义,可以说比较重要的宏才会放在gcc的D选项后边。
(1)-Dname 定义宏name,默认定义内容为字符串“1”
(2)-Dname=defn 定义宏name,并且内容为defn
例:
1.执行编译命令:
gcc test.c -o test
test的执行结果:
HELLO not define!
2.条件编译测试,执行编译命令
gcc -DHELLO test.c -o test
test的执行结果:
HELLO defined!
HELLO = 1
3.宏替换测试,执行编译命令
gcc -DHELLO=64 test.c -o test
test的执行结果:
HELLO defined!
HELLO = 64