Linux下编译C程序的过程
内容摘要
学习一门语言程序,本人觉得还是得学习它的编译规则,现在,通过小例子小结下自己对C编译的认识。
/*test.c 了解C程序的编译*/
#include <stdio.h>
int main(void)
{
print
/*test.c 了解C程序的编译*/
#include <stdio.h>
int main(void)
{
文章正文
学习一门语言程序,本人觉得还是得学习它的编译规则,现在,通过小例子小结下自己对C编译的认识。
/*test.c 了解C程序的编译*/ #include <stdio.h> int main(void) { printf("Hello World!\n"); return 0; }
对于test.c,我们常用一步编译到位的命令是:
gcc -o test test.c 或者 gcc test.c -o test
实际上,上面的这个编译命令包含了四个阶段的处理,即预处理(也称预编译,Preprocessing)、编译(Compilation)、汇编 (Assembly)和连接(Linking)。
这里详细列举完整的编译过程
预处理:
作用: 预处理的作用主要是读入源代码,检查包含预处理指令的语句和宏定义,并对源代码进行响应的转换。预处理过程还会删除程序中的注释和多余的空白字符。
对象: 预处理指令是以“#”开头的,预处理的处理对象主要包括以下方面:
(1)#define 宏定义
(2)#运算符 #运算符作用是把跟在其后的参数转换成一个字符串。
/***例***/ #define PASTE(n) "adhfkj"#n int main() { printf("%s\n",PASTE(15)); return 0; } /********输出adhfj15*********/
(3)##运算符 ##运算符的作用用于把参数连接到一起。
/*****例*****/ #define NUM(a,b,c) a##b##c #define STR(a,b,c) a##b##c int main() { printf("%d\n",NUM(1,2,3)); printf("%s\n",STR("aa","bb","cc")); return 0; } /*********最后程序的输出为:aabbcc**********/
(4)条件编译指令
(5)头文件包含指令
(6)特殊符号
__FILE__包含当前程序文件名的字符串
__LINE__表示当前行号的整数
__DATE__包含当前日期的字符串
__TIME__包含当前的字符串
如上面的test.c文件的预处理指令是
gcc -E test.c -o test.i
编译-编译成汇编语言
gcc -S test.i -o test.s
这是上面代码编译出来test.s的内容
.file "test.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 leave .cfi_def_cfa 7, 8 ret .cfi_endproc .LFE0: .size main, .-main .ident "GCC: (GNU) 4.4.7 20120313 (Red Hat 4.4.7-4)" .section .note.GNU-stack,"",@progbits
汇编
作用:将上面的汇编指令编译生成目标文件
gcc -c test.s -o test.o
这是上面的test.o文件的内容
ELF > 8 @ @ UH夊? ? ? 擅 hello world GCC: (GNU) 4.4.7 20120313 (Red Hat 4.4.7-4) zR x ? A?C P .symtab .strtab .shstrtab .rela.text .data .bss .rodata .comment .note.GNU-stack .rela.eh_frame @ ? 0 & X , X 1 X 9 0 d - B ? W ? 8 R ? ? a x
代码注释