GCC命令详解
在網上查GCC命令方面的資料,發現兩篇文章寫得比較合我意,沖突不大,特放在一起學習,希望對我及大家有益。第一篇主要講GCC原理,第二篇主要詳解GCC命令的參數。
第一篇文章:Linux GCC常用命令
1簡介
2簡單編譯
2.1預處理
2.2編譯為匯編代碼(Compilation)
2.3匯編(Assembly)
2.4連接(Linking)
3多個程序文件的編譯
4檢錯
5庫文件連接
5.1編譯成可執行文件
5.2鏈接
5.3強制鏈接時使用靜態鏈接庫
1簡介
GCC 的意思也只是 GNU C Compiler 而已。經過了這么多年的發展,GCC 已經不僅僅能支持 C 語言;它現在還支持 Ada 語言、C++ 語言、Java 語言、Objective C 語言、Pascal 語言、COBOL語言,以及支持函數式編程和邏輯編程的 Mercury 語言,等等。而 GCC 也不再單只是 GNU C 語言編譯器的意思了,而是變成了 GNU Compiler Collection 也即是 GNU 編譯器家族的意思了。另一方面,說到 GCC 對于操作系統平臺及硬件平臺支持,概括起來就是一句話:無所不在。
2簡單編譯
示例程序如下:
//test.c #include <stdio.h> int main(void) {printf("Hello World!\n");return 0; }這個程序,一步到位的編譯指令是:
gcc test.c -o test
實質上,上述編譯過程是分為四個階段進行的,即預處理(也稱預編譯,Preprocessing)、編譯(Compilation)、匯編 (Assembly)和連接(Linking)。
2.1預處理
gcc -E test.c -o test.i 或 gcc -E test.c?
可以輸出test.i文件中存放著test.c經預處理之后的代碼。打開test.i文件,看一看,就明白了。后面那條指令,是直接在命令行窗口中輸出預處理后的代碼.
gcc的-E選項,可以讓編譯器在預處理后停止,并輸出預處理結果。在本例中,預處理結果就是將stdio.h 文件中的內容插入到test.c中了。
2.2編譯為匯編代碼(Compilation)
預處理之后,可直接對生成的test.i文件編譯,生成匯編代碼:
gcc -S test.i -o test.sgcc的-S選項,表示在程序編譯期間,在生成匯編代碼后,停止,-o輸出匯編代碼文件。
2.3匯編(Assembly)
對于上一小節中生成的匯編代碼文件test.s,gas匯編器負責將其編譯為目標文件,如下:
gcc -c test.s -o test.o2.4連接(Linking)
gcc連接器是gas提供的,負責將程序的目標文件與所需的所有附加的目標文件連接起來,最終生成可執行文件。附加的目標文件包括靜態連接庫和動態連接庫。
對于上一小節中生成的test.o,將其與C標準輸入輸出庫進行連接,最終生成程序test
gcc test.o -o test?
在命令行窗口中,執行./test, 讓它說HelloWorld吧!
3多個程序文件的編譯
通常整個程序是由多個源文件組成的,相應地也就形成了多個編譯單元,使用GCC能夠很好地管理這些編譯單元。假設有一個由test1.c和 test2.c兩個源文件組成的程序,為了對它們進行編譯,并最終生成可執行程序test,可以使用下面這條命令:
gcc test1.c test2.c -o test
如果同時處理的文件不止一個,GCC仍然會按照預處理、編譯和鏈接的過程依次進行。如果深究起來,上面這條命令大致相當于依次執行如下三條命令:
gcc -c test1.c -o test1.o gcc -c test2.c -o test2.o gcc test1.o test2.o -o test
4檢錯
gcc -pedantic illcode.c -o illcode-pedantic編譯選項并不能保證被編譯程序與ANSI/ISO C標準的完全兼容,它僅僅只能用來幫助Linux程序員離這個目標越來越近。或者換句話說,-pedantic選項能夠幫助程序員發現一些不符合 ANSI/ISO C標準的代碼,但不是全部,事實上只有ANSI/ISO C語言標準中要求進行編譯器診斷的那些情況,才有可能被GCC發現并提出警告。
除了-pedantic之外,GCC還有一些其它編譯選項也能夠產生有用的警告信息。這些選項大多以-W開頭,其中最有價值的當數-Wall了,使用它能夠使GCC產生盡可能多的警告信息。
gcc -Wall illcode.c -o illcodeGCC給出的警告信息雖然從嚴格意義上說不能算作錯誤,但卻很可能成為錯誤的棲身之所。一個優秀的Linux程序員應該盡量避免產生警告信息,使自己的代碼始終保持標準、健壯的特性。所以將警告信息當成編碼錯誤來對待,是一種值得贊揚的行為!所以,在編譯程序時帶上-Werror選項,那么GCC會在所有產生警告的地方停止編譯,迫使程序員對自己的代碼進行修改,如下:
gcc -Werror test.c -o test?
5庫文件連接
開發軟件時,完全不使用第三方函數庫的情況是比較少見的,通常來講都需要借助許多函數庫的支持才能夠完成相應的功能。從程序員的角度看,函數庫實際上就是一些頭文件(.h)和庫文件(so、或lib、dll)的集合。。雖然Linux下的大多數函數都默認將頭文件放到/usr/include/目錄下,而庫文件則放到/usr/lib/目錄下;Windows所使用的庫文件主要放在Visual Stido的目錄下的include和lib,以及系統文件夾下。但也有的時候,我們要用的庫不再這些目錄下,所以GCC在編譯時必須用自己的辦法來查找所需要的頭文件和庫文件。
例如我們的程序test.c是在linux上使用c連接mysql,這個時候我們需要去mysql官網下載MySQL Connectors的C庫,下載下來解壓之后,有一個include文件夾,里面包含mysql connectors的頭文件,還有一個lib文件夾,里面包含二進制so文件libmysqlclient.so
其中inclulde文件夾的路徑是/usr/dev/mysql/include,lib文件夾是/usr/dev/mysql/lib
?
5.1編譯成可執行文件
首先我們要進行編譯test.c為目標文件,這個時候需要執行
gcc –c –I /usr/dev/mysql/include test.c –o test.o5.2鏈接
最后我們把所有目標文件鏈接成可執行文件:
gcc –L /usr/dev/mysql/lib –lmysqlclient test.o –o testLinux下的庫文件分為兩大類分別是動態鏈接庫(通常以.so結尾)和靜態鏈接庫(通常以.a結尾),二者的區別僅在于程序執行時所需的代碼是在運行時動態加載的,還是在編譯時靜態加載的。
5.3強制鏈接時使用靜態鏈接庫
默認情況下, GCC在鏈接時優先使用動態鏈接庫,只有當動態鏈接庫不存在時才考慮使用靜態鏈接庫,如果需要的話可以在編譯時加上-static選項,強制使用靜態鏈接庫。
在/usr/dev/mysql/lib目錄下有鏈接時所需要的庫文件libmysqlclient.so和libmysqlclient.a,為了讓GCC在鏈接時只用到靜態鏈接庫,可以使用下面的命令:
gcc –L /usr/dev/mysql/lib –static –lmysqlclient test.o –o test?
靜態庫鏈接時搜索路徑順序:
1. ld會去找GCC命令中的參數-L
2. 再找gcc的環境變量LIBRARY_PATH
3. 再找內定目錄 /lib /usr/lib /usr/local/lib 這是當初compile gcc時寫在程序內的
動態鏈接時、執行時搜索路徑順序:
1. 編譯目標代碼時指定的動態庫搜索路徑
2. 環境變量LD_LIBRARY_PATH指定的動態庫搜索路徑
3. 配置文件/etc/ld.so.conf中指定的動態庫搜索路徑
4. 默認的動態庫搜索路徑/lib
5. 默認的動態庫搜索路徑/usr/lib
有關環境變量:
LIBRARY_PATH環境變量:指定程序靜態鏈接庫文件搜索路徑
LD_LIBRARY_PATH環境變量:指定程序動態鏈接庫文件搜索路徑
第二篇文章:gcc 命令解釋
gcc and g++現在是gnu中最主要和最流行的c & c++編譯器 .gcc/g++在執行編譯工作的時候,總共需要以下幾步:
1.預處理,生成.i的文件[預處理器cpp]
2.將預處理后的文件不轉換成匯編語言,生成文件.s[編譯器egcs]
3.有匯編變為目標代碼(機器代碼)生成.o的文件[匯編器as]
4.連接目標代碼,生成可執行程序[鏈接器ld]
gcc and g++現在是gnu中最主要和最流行的c & c++編譯器 .gcc/g++在執行編譯工作的時候,總共需要以下幾步:
1.預處理,生成.i的文件[預處理器cpp]
2.將預處理后的文件不轉換成匯編語言,生成文件.s[編譯器egcs]
3.有匯編變為目標代碼(機器代碼)生成.o的文件[匯編器as]
4.連接目標代碼,生成可執行程序[鏈接器ld]
GCC能夠處理的后綴有:
a. *.c *.C (C語言)
b. *.cxx *.cc (C++語言)
c. *.m (面向對象的C)
d. *.i (預處理后的C語言源文件)
e. *.ii (預處理后的C++語言源文件)
f. *.s *.S (匯編語言)
h. *.h (頭文件)
目標文件可以是:
a. *.o 編譯連接后的目標文件
b. *.a 庫文件
[參數詳解]
-x language filename
設定文件所使用的語言,使后綴名無效,對以后的多個有效.也就是根據約定C語言的后
綴名稱是.c的,而C++的后綴名是.C或者.cpp,如果你很個性,決定你的C代碼文件的后綴
名是.pig 哈哈,那你就要用這個參數,這個參數對他后面的文件名都起作用,除非到了
下一個參數的使用。
可以使用的參數嗎有下面的這些
`c’, `objective-c’, `c-header’, `c++’, `cpp-output’, `assembler’, and `a
ssembler-with-cpp’.
看到英文,應該可以理解的。
例子用法:
gcc -x c hello.pig
-x none filename
關掉上一個選項,也就是讓gcc根據文件名后綴,自動識別文件類型
例子用法:
gcc -x c hello.pig -x none hello2.c
-c
只激活預處理,編譯,和匯編,也就是他只把程序做成obj文件
例子用法:
gcc -c hello.c
他將生成.o的obj文件
-S
只激活預處理和編譯,就是指把文件編譯成為匯編代碼。
例子用法
gcc -S hello.c
他將生成.s的匯編代碼,你可以用文本編輯器察看
-E
只激活預處理,這個不生成文件,你需要把它重定向到一個輸出文件里面.
例子用法:
gcc -E hello.c > pianoapan.txt
gcc -E hello.c | more
慢慢看吧,一個hello word 也要與處理成800行的代碼
-o
制定目標名稱,缺省的時候,gcc 編譯出來的文件是a.out,很難聽,如果你和我有同感
,改掉它,哈哈
例子用法
gcc -o hello.exe hello.c (哦,windows用習慣了)
gcc -o hello.asm -S hello.c
-pipe
使用管道代替編譯中臨時文件,在使用非gnu匯編工具的時候,可能有些問題
gcc -pipe -o hello.exe hello.c
-ansi
關閉gnu c中與ansi c不兼容的特性,激活ansi c的專有特性(包括禁止一些asm inl
ine typeof關鍵字,以及UNIX,vax等預處理宏,
-fno-asm
此選項實現ansi選項的功能的一部分,它禁止將asm,inline和typeof用作關鍵字。
-fno-strict-prototype
只對g++起作用,使用這個選項,g++將對不帶參數的函數,都認為是沒有顯式的對參數
的個數和類型說明,而不是沒有參數.
而gcc無論是否使用這個參數,都將對沒有帶參數的函數,認為城沒有顯式說明的類型
-fthis-is-varialble
就是向傳統c++看齊,可以使用this當一般變量使用.
-fcond-mismatch
允許條件表達式的第二和第三參數類型不匹配,表達式的值將為void類型
-funsigned-char
-fno-signed-char
-fsigned-char
-fno-unsigned-char
這四個參數是對char類型進行設置,決定將char類型設置成unsigned char(前兩個參
數)或者 signed char(后兩個參數)
-include file
包含某個代碼,簡單來說,就是便以某個文件,需要另一個文件的時候,就可以用它設
定,功能就相當于在代碼中使用#include
例子用法:
gcc hello.c -include /root/pianopan.h
-imacros file
將file文件的宏,擴展到gcc/g++的輸入文件,宏定義本身并不出現在輸入文件中
-Dmacro
相當于C語言中的#define macro
-Dmacro=defn
相當于C語言中的#define macro=defn
-Umacro
相當于C語言中的#undef macro
-undef
取消對任何非標準宏的定義
-Idir
在你是用#include”file”的時候,gcc/g++會先在當前目錄查找你所制定的頭文件,如
果沒有找到,他回到缺省的頭文件目錄找,如果使用-I制定了目錄,他
回先在你所制定的目錄查找,然后再按常規的順序去找.
對于#include,gcc/g++會到-I制定的目錄查找,查找不到,然后將到系統的缺
省的頭文件目錄查找
-I-
就是取消前一個參數的功能,所以一般在-Idir之后使用
-idirafter dir
在-I的目錄里面查找失敗,講到這個目錄里面查找.
-iprefix prefix
-iwithprefix dir
一般一起使用,當-I的目錄查找失敗,會到prefix+dir下查找
-nostdinc
使編譯器不再系統缺省的頭文件目錄里面找頭文件,一般和-I聯合使用,明確限定頭
文件的位置
-nostdin C++
規定不在g++指定的標準路經中搜索,但仍在其他路徑中搜索,.此選項在創libg++庫
使用
-C
在預處理的時候,不刪除注釋信息,一般和-E使用,有時候分析程序,用這個很方便的
-M
生成文件關聯的信息。包含目標文件所依賴的所有源代碼你可以用gcc -M hello.c
來測試一下,很簡單。
-MM
和上面的那個一樣,但是它將忽略由#include造成的依賴關系。
-MD
和-M相同,但是輸出將導入到.d的文件里面
-MMD
和-MM相同,但是輸出將導入到.d的文件里面
-Wa,option
此選項傳遞option給匯編程序;如果option中間有逗號,就將option分成多個選項,然
后傳遞給會匯編程序
-Wl.option
此選項傳遞option給連接程序;如果option中間有逗號,就將option分成多個選項,然
后傳遞給會連接程序.
-llibrary
制定編譯的時候使用的庫
例子用法
gcc -lcurses hello.c
使用ncurses庫編譯程序
-Ldir
制定編譯的時候,搜索庫的路徑。比如你自己的庫,可以用它制定目錄,不然
編譯器將只在標準庫的目錄找。這個dir就是目錄的名稱。
-O0
-O1
-O2
-O3
編譯器的優化選項的4個級別,-O0表示沒有優化,-O1為缺省值,-O3優化級別最高
-g
只是編譯器,在編譯的時候,產生調試信息。
-gstabs
此選項以stabs格式聲稱調試信息,但是不包括gdb調試信息.
-gstabs+
此選項以stabs格式聲稱調試信息,并且包含僅供gdb使用的額外調試信息.
-ggdb
此選項將盡可能的生成gdb的可以使用的調試信息.
-static
此選項將禁止使用動態庫,所以,編譯出來的東西,一般都很大,也不需要什么
動態連接庫,就可以運行.
-share
此選項將盡量使用動態庫,所以生成文件比較小,但是需要系統由動態庫.
-traditional
試圖讓編譯器支持傳統的C語言特性
[參考資料]
-Linux/UNIX高級編程
中科紅旗軟件技術有限公司編著.清華大學出版社出版
-Gcc man page
[ChangeLog]
-2002-08-10
ver 0.1 發布最初的文檔
-2002-08-11
ver 0.11 修改文檔格式
-2002-08-12
ver 0.12 加入了對靜態庫,動態庫的參數
-2002-08-16
ver 0.16 增加了gcc編譯的4個階段的命令
運行 gcc/egcs
**********運行 gcc/egcs***********************
GCC 是 GNU 的 C 和 C++ 編譯器。實際上,GCC 能夠編譯三種語言:C、C++ 和 O
bject C(C 語言的一種面向對象擴展)。利用 gcc 命令可同時編譯并連接 C 和 C++
源程序。
如果你有兩個或少數幾個 C 源文件,也可以方便地利用 GCC 編譯、連接并生成可
執行文件。例如,假設你有兩個源文件 main.c 和 factorial.c 兩個源文件,現在要編
譯生成一個計算階乘的程序。
代碼:
———————–
清單 factorial.c
———————–
int factorial (int n)
{
if (n
#include
int factorial (int n);
int main (int argc, char **argv)
{
int n;
if (argc
void main (void)
{
cout
}
?
gcc and g++分別是gnu的c & c++編譯器?
-x language filename?
設定文件所使用的語言,使后綴名無效,對以后的多個有效.也就是根?
據約定C語言的后綴名稱是.c的,而C++的后綴名是.C或者.cpp,如果?
你很個性,決定你的C代碼文件的后綴名是.pig 哈哈,那你就要用這?
個參數,這個參數對他后面的文件名都起作用,除非到了下一個參數?
的使用。?
可以使用的參數嗎有下面的這些?
`c', `objective-c', `c-header', `c++', `cpp-output',?
`assembler', and `assembler-with-cpp'.?
看到英文,應該可以理解的。?
例子用法:?
gcc -x c hello.pig?
?
-x none filename?
關掉上一個選項,也就是讓gcc根據文件名后綴,自動識別文件類型?
例子用法:?
gcc -x c hello.pig -x none hello2.c?
?
-c?
只激活預處理,編譯,和匯編,也就是他只把程序做成obj文件?
例子用法:?
gcc -c hello.c?
他將生成.o的obj文件?
-S?
只激活預處理和編譯,就是指把文件編譯成為匯編代碼。?
例子用法?
gcc -S hello.c?
他將生成.s的匯編代碼,你可以用文本編輯器察看?
-E?
只激活預處理,這個不生成文件,你需要把它重定向到一個輸出文件里?
面.?
例子用法:?
gcc -E hello.c > pianoapan.txt?
gcc -E hello.c | more?
慢慢看吧,一個hello word 也要與處理成800行的代碼?
-o?
制定目標名稱,缺省的時候,gcc 編譯出來的文件是a.out,很難聽,如果?
你和我有同感,改掉它,哈哈?
例子用法?
gcc -o hello.exe hello.c (哦,windows用習慣了)?
gcc -o hello.asm -S hello.c?
-pipe?
使用管道代替編譯中臨時文件,在使用非gnu匯編工具的時候,可能有些問?
題?
gcc -pipe -o hello.exe hello.c?
-ansi?
關閉gnu c中與ansi c不兼容的特性,激活ansi c的專有特性(包括禁止一?
些asm inline typeof關鍵字,以及UNIX,vax等預處理宏,?
-fno-asm?
此選項實現ansi選項的功能的一部分,它禁止將asm,inline和typeof用作?
關鍵字。?
?
-fno-strict-prototype?
只對g++起作用,使用這個選項,g++將對不帶參數的函數,都認為是沒有顯式?
的對參數的個數和類型說明,而不是沒有參數.?
而gcc無論是否使用這個參數,都將對沒有帶參數的函數,認為城沒有顯式說?
明的類型?
?
-fthis-is-varialble?
就是向傳統c++看齊,可以使用this當一般變量使用.?
?
-fcond-mismatch?
允許條件表達式的第二和第三參數類型不匹配,表達式的值將為void類型?
?
-funsigned-char?
-fno-signed-char?
-fsigned-char?
-fno-unsigned-char?
這四個參數是對char類型進行設置,決定將char類型設置成unsigned char(前?
兩個參數)或者 signed char(后兩個參數)?
?
-include file?
包含某個代碼,簡單來說,就是便以某個文件,需要另一個文件的時候,就可以?
用它設定,功能就相當于在代碼中使用#include?
例子用法:?
gcc hello.c -include /root/pianopan.h?
?
-imacros file?
將file文件的宏,擴展到gcc/g++的輸入文件,宏定義本身并不出現在輸入文件?
中?
?
-Dmacro?
相當于C語言中的#define macro?
?
-Dmacro=defn?
相當于C語言中的#define macro=defn?
?
-Umacro?
相當于C語言中的#undef macro?
-undef?
取消對任何非標準宏的定義?
?
-Idir?
在你是用#include"file"的時候,gcc/g++會先在當前目錄查找你所制定的頭?
文件,如果沒有找到,他回到缺省的頭文件目錄找,如果使用-I制定了目錄,他?
回先在你所制定的目錄查找,然后再按常規的順序去找.?
對于#include,gcc/g++會到-I制定的目錄查找,查找不到,然后將到系?
統的缺省的頭文件目錄查找?
?
-I-?
就是取消前一個參數的功能,所以一般在-Idir之后使用?
?
-idirafter dir?
在-I的目錄里面查找失敗,講到這個目錄里面查找.?
?
-iprefix prefix?
-iwithprefix dir?
一般一起使用,當-I的目錄查找失敗,會到prefix+dir下查找?
?
-nostdinc?
使編譯器不再系統缺省的頭文件目錄里面找頭文件,一般和-I聯合使用,明確?
限定頭文件的位置?
?
-nostdin C++?
規定不在g++指定的標準路經中搜索,但仍在其他路徑中搜索,.此選項在創建?
libg++庫使用?
?
-C?
在預處理的時候,不刪除注釋信息,一般和-E使用,有時候分析程序,用這個很?
方便的?
?
-M?
生成文件關聯的信息。包含目標文件所依賴的所有源代碼?
你可以用gcc -M hello.c來測試一下,很簡單。?
?
-MM?
和上面的那個一樣,但是它將忽略由#include造成的依賴關系。?
?
-MD?
和-M相同,但是輸出將導入到.d的文件里面?
?
-MMD?
和-MM相同,但是輸出將導入到.d的文件里面?
?
-Wa,option?
此選項傳遞option給匯編程序;如果option中間有逗號,就將option分成多個選?
項,然后傳遞給會匯編程序?
?
-Wl.option?
此選項傳遞option給連接程序;如果option中間有逗號,就將option分成多個選?
項,然后傳遞給會連接程序.?
?
-llibrary?
制定編譯的時候使用的庫?
例子用法?
gcc -lcurses hello.c?
使用ncurses庫編譯程序?
?
-Ldir?
制定編譯的時候,搜索庫的路徑。比如你自己的庫,可以用它制定目錄,不然?
編譯器將只在標準庫的目錄找。這個dir就是目錄的名稱。?
?
-O0?
-O1?
-O2?
-O3?
編譯器的優化選項的4個級別,-O0表示沒有優化,-O1為缺省值,-O3優化級別最?
高 ?
?
-g?
只是編譯器,在編譯的時候,產生條是信息。?
?
-gstabs?
此選項以stabs格式聲稱調試信息,但是不包括gdb調試信息.?
?
-gstabs+?
此選項以stabs格式聲稱調試信息,并且包含僅供gdb使用的額外調試信息.?
?
-ggdb?
此選項將盡可能的生成gdb的可以使用的調試信息.?
-static?
此選項將禁止使用動態庫,所以,編譯出來的東西,一般都很大,也不需要什么?
動態連接庫,就可以運行.?
-share?
此選項將盡量使用動態庫,所以生成文件比較小,但是需要系統由動態庫.?
-traditional?
試圖讓編譯器支持傳統的C語言特性
總結
- 上一篇: 图灵 计算机 ppt,turing ma
- 下一篇: 力扣 706. 设计哈希映射