學習一門語言程序,本人以爲仍是得學習它的編譯規則,如今,經過小例子小結下本身對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文件就是最終系統能夠執行的文件。
對於程序的編譯,咱們通常把它認爲「編譯」和「連接」兩部分也足夠了,這裏的編譯已經包括了預處理,編譯成彙編語言和編譯成目標文件三個步驟了。只要頭文件完整,語法無誤,編譯通常都能經過。只要有完整的目標文件和功能庫文件,連接也能夠成功。只要編譯經過了,連接也經過了,整個項目的編譯就算完成了。