Linux 下編譯C程序的全過程


學習一門語言程序,本人覺得還是得學習它的編譯規則,現在,通過小例子小結下自己對C編譯的認識。

?
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,我們常用一步編譯到位的命令是:

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

實際上,上面的這個編譯命令包含了四個階段的處理,即預處理(也稱預編譯,Preprocessing)、編譯(Compilation)、匯編 (Assembly)和連接(Linking)。

這里詳細列舉完整的編譯過程

預處理:

作用:  預處理的作用主要是讀入源代碼,檢查包含預處理指令的語句和宏定義,並對源代碼進行響應的轉換。預處理過程還會刪除程序中的注釋和多余的空白字符。

對象:  預處理指令是以“#”開頭的,預處理的處理對象主要包括以下方面:

  (1)#define  宏定義

  (2)#運算符    #運算符作用是把跟在其后的參數轉換成一個字符串。     

?
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文件就是最終系統可以執行的文件。

對於程序的編譯,我們一般把它認為“編譯”和“鏈接”兩部分也足夠了,這里的編譯已經包括了預處理,編譯成匯編語言和編譯成目標文件三個步驟了。只要頭文件完整,語法無誤,編譯一般都能通過。只要有完整的目標文件和功能庫文件,鏈接也可以成功。只要編譯通過了,鏈接也通過了,整個項目的編譯就算完成了。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM