#gcc hello.c
該命令將hello.c直接生成最終二進制可執行程序a.out
這條命令隱含執行了(1)預處理、(2)匯編、(3)編譯並(4)鏈接形成最終的二進制可執行程序。這里未指定輸出文件,默認輸出為a.out。
從上面我們知道GCC編譯源代碼生成最終可執行的二進制程序,GCC后台隱含執行了四個階段步驟。
GCC編譯C源碼有四個步驟:
預處理-----> 編譯 ----> 匯編 ----> 鏈接
現在我們就用GCC的命令選項來逐個剖析GCC過程。
1)預處理(Pre-processing)
在該階段,編譯器將C源代碼中的包含的頭文件如stdio.h編譯進來,用戶可以使用gcc的選項”-E”進行查看。
用法:#gcc -E hello.c -o hello.i
作用:將hello.c預處理輸出hello.i文件。
2)編譯階段(Compiling)
第二步進行的是編譯階段,在這個階段中,Gcc首先要檢查代碼的規范性、是否有語法錯誤等,以確定代碼的實際要做的工作,在檢查無誤后,Gcc把代碼翻譯 成匯編語言。用戶可以使用”-S”選項來進行查看,該選項只進行編譯而不進行匯編,生成匯編代碼。
選項 -S
用法:[root]# gcc –S hello.i –o hello.s
作用:將預處理輸出文件hello.i匯編成hello.s文件。
[root@richard hello-gcc]# ls
hello.c hello.i hello.s
3)匯編階段(Assembling)
匯編階段是把編譯階段生成的”.s”文件轉成二進制目標代碼.
選項 -c
用法:[root]# gcc –c hello.s –o hello.o
作用:將匯編輸出文件test.s編譯輸出test.o文件。
[root]# gcc -c hello.s -o hello.o
[root]# ls
hello.c hello.i hello.o hello.s
4)鏈接階段(Link)
在成功編譯之后,就進入了鏈接階段。
無選項鏈接
用法:[root]# gcc hello.o –o hello.exe
作用:將編譯輸出文件hello.o鏈接成最終可執行文件hello.exe。
[root]# ls
hello.c hello.exe hello.i hello.o hello.s
運行該可執行文件,出現正確的結果如下。
[root@localhost Gcc]# ./hello
Hello World!
在這里涉及到一個重要的概念:函數庫。
讀者可以重新查看這個小程序,在這個程序中並沒有定義”printf”的函數實現,且在預編譯中包含進 的”stdio.h”中也只有該函數的聲明,而沒有定義函數的實現,那么,是在哪里實現”printf”函數的呢?最后的答案是:系統把這些函數實現都被 做到名為libc.so.6的庫文件中去了,在沒有特別指定時,gcc會到系統默認的搜索路徑”/usr/lib”下進行查找,也就是鏈接到 libc.so.6庫函數中去,這樣就能實現函數”printf” 了,而這也就是鏈接的作用。
你可以用ldd命令查看動態庫加載情況:
[root]# ldd hello.exe
libc.so.6 => /lib/tls/libc.so.6 (0x42000000)
/lib/ld-linux.so.2 => /lib/ld-linux.so.2 (0x40000000)
函數庫一般分為靜態庫和動態庫兩種。靜態庫是指編譯鏈接時,把庫文件的代碼全部加入到可執行文件中,因此生成的文件比較大,但在運行時也就不再需要 庫文件了。其后綴名一般為”.a”。動態庫與之相反,在編譯鏈接時並沒有把庫文件的代碼加入到可執行文件中,而是在程序執行時由運行時鏈接文件加載庫,這 樣可以節省系統的開銷。動態庫一般后綴名為”.so”,如前面所述的libc.so.6就是動態庫。gcc在編譯時默認使用動態庫。