源文件到可执行文件的转换流程:
源文件->预处理->编译->汇编->链接
--------------------------------------------------------------------------------------------------------------------------
[参数详解]
-x language filename
设定文件所使用的语言,使后缀名无效,对以后的多个有效.也就是根据约定C语言的后缀名称是.c
的,而C++的后缀名是.C
或者.cpp
,如果你很个性,决定你的C代码文件的后缀名是.pig 哈哈,那你就要用这个参数,这个参数对他后面的文件名都起作用,除非到了
下一个参数的使用。
可以使用的参数有下面的这些
c',
objective-c', c-header',
c++', cpp-output',
assembler', and `a
ssembler-with-cpp'.
看到英文,应该可以理解的。
例子用法:
gcc -x c hello.pig
-x none filename
关掉上一个选项,也就是让gcc根据文件名后缀,自动识别文件类型
例子用法:
gcc -x c hello.pig -x none hello2.c
-----------------------------------------------------------------------------------------------------------------------
-c
只激活预处理,编译,和汇编,也就是他只把程序做成obj文件
例子用法:
gcc -c hello.c
他将生成.o的obj文件
--------------------------------------------------------------------------------------------------------------------------
-S
只激活预处理和编译,就是指把文件编译成为汇编代码。
例子用法
gcc -S hello.c
他将生成.s的汇编代码,你可以用文本编辑器察看
--------------------------------------------------------------------------------------------------------------------------
-E
只激活预处理,这个不生成文件,你需要把它重定向到一个输出文件里面.
例子用法:
gcc -E hello.c > pianoapan.txt
gcc -E hello.c | more
慢慢看吧,一个hello word 也要与处理成800行的代码
--------------------------------------------------------------------------------------------------------------------------
-o
制定目标名称,缺省的时候,gcc 编译出来的文件是a.out,很难听,如果你和我有同感
,改掉它,哈哈
例子用法
gcc -o hello.exe hello.c (哦,windows用习惯了)
gcc -o hello.asm -S hello.c
--------------------------------------------------------------------------------------------------------------------------
-pipe
使用管道代替编译中临时文件,在使用非gnu汇编工具的时候,可能有些问题
gcc -pipe -o hello.exe hello.c
--------------------------------------------------------------------------------------------------------------------------
-ansi
关闭gnu c中与ansi c不兼容的特性,激活ansi c的专有特性(包括禁止一些asm inl
ine typeof关键字,以及UNIX,vax等预处理宏
--------------------------------------------------------------------------------------------------------------------------
--------------------------------------------------------------------------------------------------------------------------
参考程序:hello.c
#include<stdio.h>
int main()
{
printf("Hello World
");
return 0;
}
1. 预处理:gcc -o hello.i -E hello.c 或cpp -o hello.i hello.c
编译器cpp(the C Preprocessor)把以#开头的代码进行预处理,即替换;比如hello.c中的第一行为#include<stdio.h>,预处理器便将stdio.h的内容直接插入到程序中,可以看到代码从6行扩展到854行。
823 extern FILE *popen (const char *__command, const char *__modes) ;
824
825
826
827
828
829 extern int pclose (FILE *__stream);
830
831
832
833
834
835 extern char *ctermid (char *__s) __attribute__ ((__nothrow__ , __leaf__));
836 # 912 "/usr/include/stdio.h" 3 4
837 extern void flockfile (FILE *__stream) __attribute__ ((__nothrow__ , __leaf__));
838
839
840
841 extern int ftrylockfile (FILE *__stream) __attribute__ ((__nothrow__ , __leaf__)) ;
842
843
844 extern void funlockfile (FILE *__stream) __attribute__ ((__nothrow__ , __leaf__));
845 # 942 "/usr/include/stdio.h" 3 4
846
847 # 2 "hello.c" 2
848
849 # 2 "hello.c"
850 int main()
851 {
852 printf("Hello World");
853 return 0;
854 }
2. 编译:gcc -o hello.s -S hello.i或 ccl -o hello.s hello.i
编译器ccl将文本文件hello.i编译成hello.s,这个文件里面包含一个汇编程序
.file "hello.c"
.section .rodata
.LC0:
.string "Hello World"
.text
.globl main
.type main, @function
main:
.LFB0:
.cfi_startproc
pushq %rbp
.cfi_def_cfa_offset 16
.cfi_offset 6, -16
movq %rsp, %rbp
.cfi_def_cfa_register 6
movl $.LC0, %edi
call puts
movl $0, %eax
popq %rbp
.cfi_def_cfa 7, 8
ret
.cfi_endproc
.LFE0:
.size main, .-main
.ident "GCC: (Ubuntu 5.4.0-6ubuntu1~16.04.4) 5.4.0 20160609"
.section .note.GNU-stack,"",@progbits
~
3. 汇编阶段:gcc -o hello.o -c hello.s 或 as -o hello.o hello.s
汇编器as将hello.s翻译成机器语言保留在hello.o中,这个就是二进制文件,可以用hexdump hello.o打开(hexdump 16进制查看器),部分如下:
0000010 0001 003e 0001 0000 0000 0000 0000 0000
0000020 0000 0000 0000 0000 02a0 0000 0000 0000
0000030 0000 0000 0040 0000 0000 0040 000d 000a
0000040 4855 e589 00bf 0000 e800 0000 0000 00b8
0000050 0000 5d00 48c3 6c65 6f6c 5720 726f 646c
0000060 0000 4347 3a43 2820 6255 6e75 7574 3520
0000070 342e 302e 362d 6275 6e75 7574 7e31 3631
0000080 302e 2e34 2934 3520 342e 302e 3220 3130
0000090 3036 3036 0039 0000 0014 0000 0000 0000
00000a0 7a01 0052 7801 0110 0c1b 0807 0190 0000
00000b0 001c 0000 001c 0000 0000 0000 0015 0000
00000c0 4100 100e 0286 0d43 5006 070c 0008 0000
00000d0 0000 0000 0000 0000 0000 0000 0000 0000
00000e0 0000 0000 0000 0000 0001 0000 0004 fff1
00000f0 0000 0000 0000 0000 0000 0000 0000 0000
4. 链接:gcc -o hello hello.o 或 ld -o hello hello.o
hello.c调用了printf函数,而printf是标准C库的一个函数,他保存在一个名为printf.o 的文件中,这个文件必须以某种方式合并到我们的hello.o的程序中
编译器ld就是负责这种合并,结果得到hello可执行文件,可以被加载到到内存中由系统执行
最后执行:./hello
Hello World
5.一步命令编译链接:
gcc -o hello hello.c
./hello
--------------------------------------------------------------------------------------------------------------------------
C++程序的编译
GCC 可同时用来编译 C 程序和 C++ 程序。一般来说,C 编译器通过源文件的后缀
名来判断是 C 程序还是 C++ 程序。在 Linux 中,C 源文件的后缀名为 .c,而 C++ 源
文件的后缀名为 .C 或 .cpp。但是,gcc 命令只能编译 C++ 源文件,而不能自动和 C
++ 程序使用的库连接。因此,通常使用 g++ 命令来完成 C++ 程序的编译和连接,该程序会自动调用 gcc 实现编译,具体的命令与 gcc 基本一致
--------------------------------------------------------------------------------------------------------------------------
参考:
GCC编译过程简述
GCC 参数详解