歡迎來到Linux教程網
Linux教程網
Linux教程網
Linux教程網
Linux教程網 >> Linux基礎 >> Linux教程 >> Linux下GCC使用方法簡介

Linux下GCC使用方法簡介

日期:2017/2/28 16:07:11   编辑:Linux教程

編譯:
第一步、是進行預編譯,使用-E參數可以讓GCC在預處理結束後停止編譯過程:
gcc -E hello.c -o hello.i
預處理的宏定義插入到hello.i中

第二步、是將hello.i編譯為目標代碼,這可以通過使用-c參數來完成:
gcc -c hello.i -o hello.o
也可以通過源文件直接生成
gcc -c hello.c

第三步、是將生成的目標文件鏈接成可執行文件:
gcc hello.o -o hello
也可以通過源文件直接生成
gcc -o hello hello.c

警告:
1、-pedantic 選項,那麼使用了擴展語法的地方將產生相應的警告信息
2、-Wall 使用它能夠使GCC產生盡可能多的警告信息
3、-Werror,它要求GCC將所有的警告當成錯誤進行處理

庫依賴:
1、Linux下的大多數函數都默認:
頭文件放到/usr/include/目錄下
而庫文件則放到/usr/lib/目錄下
2、GCC在編譯時必須有自己的辦法來查找所需要的頭文件和庫文件。
-I選項可以向GCC的頭文件搜索路徑中添加新的目錄。
例如,如果在/home/yy/include/目錄下有編譯時所需要的頭文件,為了讓GCC能夠順利地找到它們,就可以使用-I選項:
gcc -o test test.c -I /home/yy/include

-L選項向GCC的庫文件搜索路徑中添加新的目錄
例如,如果在/home/yy/lib/目錄下有鏈接時所需要的庫文件libfoo.so,為了讓GCC能夠順利地找到它,可以使用下面的命令
gcc -o test test.c -L /home/yy/lib -lfoo

值得好好解釋一下的是-l選項,它指示GCC去連接庫文件libfoo.so。
Linux下的庫文件在命名時有一個約定,那就是應該以lib三個字母開頭,由於所有的庫文件都遵循了同樣的規范,因此在用-l選項指定鏈接的庫文件名時可以省去lib三個字母,也就是說GCC在對-lfoo進行處理時,會自動去鏈接名為libfoo.so的文件。

-static選項,強制使用靜態鏈接庫
如果在/home/yy/lib/目錄下有鏈接時所需要的庫文件libfoo.so和libfoo.a
為了讓GCC在鏈接時只用到靜態鏈接庫,可以使用下面的命令:
gcc -o test test.c -L /home/yy/lib -static -lfoo

編譯成動態庫:

g++ -fPIC -shared -o libfunc.so func.cpp

-fPIC參數
表示編譯為位置獨立的代碼,用於編譯共享庫。目標文件需要創建成位置無關碼,概念上就是在可執行程序裝載它們的時候,它們可以放在可執行程序的內存裡的任何地方。


選項-O可以告訴GCC同時減小代碼的長度和執行時間,其效果等價於-O1。
在這一級別上能夠進行的優化類型雖然取決於目標處理器,但一般都會包括線程跳轉(Thread Jump)和延遲退棧(Deferred Stack Pops)兩種優化。

選項-O2告訴GCC除了完成所有-O1級別的優化之外,同時還要進行一些額外的調整工作,如處理器指令調度等。

選項-O3則除了完成所有-O2級別的優化之外,還包括循環展開和其它一些與處理器特性相關的優化工作。

通常來說,數字越大優化的等級越高,同時也就意味著程序的運行速度越快。

許多Linux程序員都喜歡使用-O2選項,因為它在優化長度、編譯時間和代碼大小之間,取得了一個比較理想的平衡點。

time ./test 查看程序執行時間

優化雖然能夠給程序帶來更好的執行性能,但在如下一些場合中應該避免優化代碼:

程序開發的時候優化等級越高,消耗在編譯上的時間就越長,因此在開發的時候最好不要使用優化選項,只有到軟件發行或開發結束的時候,才考慮對最終生成的代碼進行優化。

資源受限的時候一些優化選項會增加可執行代碼的體積,如果程序在運行時能夠申請到的內存資源非常緊張(如一些實時嵌入式設備),那就不要對代碼進行優化,因為由這帶來的負面影響可能會產生非常嚴重的後果。

跟蹤調試的時候在對代碼進行優化的時候,某些代碼可能會被刪除或改寫,或者為了取得更佳的性能而進行重組,從而使跟蹤和調試變得異常困難。
加速:
使用管道代替編譯中臨時文件,
-pipe 加速編譯
gcc -pipe foo.c -o foo

GCC常用選項
-c 通知GCC取消鏈接步驟,即編譯源碼並在最後生成目標文件;

-Dmacro 定義指定的宏,使它能夠通過源碼中的#ifdef進行檢驗;

-E 不經過編譯預處理程序的輸出而輸送至標准輸出;

-g3 獲得有關調試程序的詳細信息,它不能與-o選項聯合使用;

-Idirectory 在包含文件搜索路徑的起點處添加指定目錄;

-llibrary 提示鏈接程序在創建最終可執行文件時包含指定的庫;

-O、-O2、-O3 將優化狀態打開,該選項不能與-g選項聯合使用;

-S 要求編譯程序生成來自源代碼的匯編程序輸出;

-v 啟動所有警報;

-Wall 在發生警報時取消編譯操作,即將警報看作是錯誤;

-Werror 在發生警報時取消編譯操作,即把報警當作是錯誤;

-w 禁止所有的報警。

靜態鏈接

  在這種連接方式下,函數的代碼將直接拷貝到最終的可執行文件中。該程序被執行時候,會被裝入該進程的虛擬地址空間中。靜態鏈接庫實際上是一個或若干目標文件。

動態鏈接

  這種方式下,函數的代碼被放到稱作動態連接庫或共享對象的某個目標文件中。鏈接程序此時的工作只是在生成的可執行文件中,記錄下共享對象的名字以及少量關鍵信息。動態連接庫可以被多個進程共享,在運行時候內存中只有一個實例。

二者比較

  使用動態鏈接能夠使可執行文件較小,並且當共享對象被多個進程使用時節省內存。但有時候系統運行改變或升級,不能保證動態連接庫一定可用、有效。

Copyright © Linux教程網 All Rights Reserved