用gcc編譯成可執行程序 (轉)


#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在編譯時默認使用動態庫。


免責聲明!

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



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