Linux 下编译C程序的全过程

学习一门语言程序,本人以为仍是得学习它的编译规则,如今,经过小例子小结下本身对C编译的认识。bash

?
1
2
3
4
5
6
7
8
/*test.c   了解C程序的编译*/
 
#include <stdio.h>
int main( void )
{
  printf ( "Hello World!\n" );
  return 0;
}

对于test.c,咱们经常使用一步编译到位的命令是:ide

?
1
gcc -o test test .c 或者 gcc test .c -o test

实际上,上面的这个编译命令包含了四个阶段的处理,即预处理(也称预编译,Preprocessing)、编译(Compilation)、汇编 (Assembly)和链接(Linking)。学习

这里详细列举完整的编译过程spa

预处理:.net

做用:  预处理的做用主要是读入源代码,检查包含预处理指令的语句和宏定义,并对源代码进行响应的转换。预处理过程还会删除程序中的注释和多余的空白字符。code

对象:  预处理指令是以“#”开头的,预处理的处理对象主要包括如下方面:htm

  (1)#define  宏定义对象

  (2)#运算符    #运算符做用是把跟在其后的参数转换成一个字符串。     ci

?
1
2
3
4
5
6
7
8
/***例***/
#define PASTE(n) "adhfkj"#n
int main()
{
    printf ( "%s\n" ,PASTE(15));
    return 0;
}
/********输出adhfj15*********/

  (3)##运算符  ##运算符的做用用于把参数链接到一块儿。 字符串

?
1
2
3
4
5
6
7
8
9
10
   /*****例*****/
  #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文件的预处理指令是

?
1
gcc -E test .c -o test .i

 

编译-编译成汇编语言

?
1
gcc -S test .i -o test .s

这是上面代码编译出来test.s的内容

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
.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

汇编

做用:将上面的汇编指令编译生成目标文件

?
1
gcc -c test .s -o test .o

这是上面的test.o文件的内容

?
1
2
3
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     €     ?                    test.c main puts 
  ?  

连接

连接的主要目的是将程序的目标文件与所须要附加的目标文件连接起来,最终生成可执行文件。附加的目标文件也包括了所须要的库文件(静态连接库和动态连接库)

?
1
gcc test .o -o test

最终生成的test文件就是最终系统能够执行的文件。

对于程序的编译,咱们通常把它认为“编译”和“连接”两部分也足够了,这里的编译已经包括了预处理,编译成汇编语言和编译成目标文件三个步骤了。只要头文件完整,语法无误,编译通常都能经过。只要有完整的目标文件和功能库文件,连接也能够成功。只要编译经过了,连接也经过了,整个项目的编译就算完成了。

相关文章
相关标签/搜索