第一篇文章:linux GCC常用命令
1簡介
2簡單編譯
2.1預處理
2.2編譯為匯編代碼(Compilation)
2.3匯編(Assembly)
2.4連接(Linking)
3多個程序文件的編譯
4檢錯
5庫文件連接
5.1編譯成可執行文件
5.2鏈接
5.3強制鏈接時使用靜態鏈接庫
GCC 的意思也只是 GNU C Compiler 而已。經過了這么多年的發展,GCC 已經不僅僅能支持 C 語言;它現在還支持 Ada 語言、C++ 語言、java 語言、Objective C 語言、Pascal 語言、COBOL語言,以及支持函數式編程和邏輯編程的 Mercury 語言,等等。而 GCC 也不再單只是 GNU C 語言編譯器的意思了,而是變成了 GNU Compiler Collection 也即是 GNU 編譯器家族的意思了。另一方面,說到 GCC 對于操作系統平臺及硬件平臺支持,概括起來就是一句話:無所不在。
示例程序如下:

//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.ogcc -c test2.c -o test2.ogcc test1.o test2.o -o test4檢錯
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 test5庫文件連接
開發軟件時,完全不使用第三方函數庫的情況是比較少見的,通常來講都需要借助許多函數庫的支持才能夠完成相應的功能。從程序員的角度看,函數庫實際上就是一些頭文件(.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命令中的參數-L2. 再找gcc的環境變量LIBRARY_PATH3. 再找內定目錄 /lib /usr/lib /usr/local/lib 這是當初compile gcc時寫在程序內的
動態鏈接時、執行時搜索路徑順序:
1. 編譯目標代碼時指定的動態庫搜索路徑2. 環境變量LD_LIBRARY_PATH指定的動態庫搜索路徑3. 配置文件/etc/ld.so.conf中指定的動態庫搜索路徑4. 默認的動態庫搜索路徑/lib5. 默認的動態庫搜索路徑/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 `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語言特性[參考資料]-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++ 和 Object 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 (argcvoid 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語言特性
轉自:點擊打開鏈接完!
新聞熱點
疑難解答