gcc命令列詳解

shiri512003發表於2010-03-27
[@more@]

本文轉自

版權歸原文作者所有

gcc命令列詳解

1、gcc包含的c/c++編譯器

gcc、cc、c++、g++
gcc和cc是一樣的,c++和g++是一樣的,一般c程式就用gcc編譯,c++程式就用g++編譯

2、gcc的基本用法

gcc test.c這樣將編譯出一個名為a.out的程式
gcc test.c -o test這樣將編譯出一個名為test的程式
-o引數用來指定生成程式的名字

3、為什麼會出現undefined reference to 'xxxxx'錯誤?

首先這是連結錯誤,不是編譯錯誤,也就是說如果只有這個錯誤,說明你的程式原始碼本身沒有問題,是你用編譯器編譯時引數用得不對,你沒有指定連結程式要用到得庫,比如你的程式裡用到了一些數學函式,那麼你就要在編譯引數裡指定程式要連結數學庫,方法是在編譯命令列里加入-lm

4、-l引數和-L引數

-l引數就是用來指定程式要連結的庫,-l引數緊接著就是庫名,那麼庫名跟真正的庫檔名有什麼關係呢?就拿數學庫來說,他的庫名是m,他的庫檔名是libm.so,很容易看出,把庫檔名的頭lib和尾.so去掉就是庫名了

好了現在我們知道怎麼得到庫名,當我們自已要用到一個第三方提供的庫名字libtest.so,那麼我們只要把libtest.so複製到 /usr/lib裡,編譯時加上-ltest引數,我們就能用上libtest.so庫了(當然要用libtest.so庫裡的函式,我們還需要與 libtest.so配套的標頭檔案)

放在/lib和/usr/lib和/usr/local/lib裡的庫直接用-l引數就能連結了,但如果庫檔案沒放在這三個目錄裡,而是放在其他目錄裡,這時我們只用-l引數的話,連結還是會出錯,出錯資訊大概是:“/usr/bin/ld: cannot find -lxxx”,也就是連結程式ld在那3個目錄裡找不到libxxx.so,這時另外一個引數-L就派上用場了,比如常用的X11的庫,它在 /usr/X11R6/lib目錄下,我們編譯時就要用-L/usr/X11R6/lib -lX11引數,-L引數跟著的是庫檔案所在的目錄名。再比如我們把libtest.so放在/aaa/bbb/ccc目錄下,那連結引數就是- L/aaa/bbb/ccc -ltest

另外,大部分libxxxx.so只是一個連結,以RH9為例,比如libm.so它連結到/lib/libm.s
o.x,/lib/libm.so.6又連結到/lib/libm-2.3.2.so,

如果沒有這樣的連結,還是會出錯,因為ld只會找libxxxx.so,所以如果你要用到xxxx
庫,而只有libxxxx.so.x或者libxxxx-x.x.x.so,做一個連結就可以了
ln -s libxxxx-x.x.x.so libxxxx.so

手工來寫連結引數總是很麻煩的,還好很多庫開發包提供了生成連結引數的程式,名字一般叫xxxx-config,一般放在/usr/bin目錄下,比如

gtk1.2的連結引數生成程式是gtk-config,執行gtk-config --libs就能得到以下輸出"-
L/usr/lib -L/usr/X11R6/lib -lgtk -lgdk -rdynamic

-lgmodule -lglib -ldl -lXi -lXext -lX11 -lm",這就是編譯一個gtk1.2程式所需的g
tk連結引數,xxx-config除了--libs引數外還有一個引數是--cflags用來生成頭文
件包含目錄的,也就是-I引數,在下面我們將會講到。你可以試試執行gtk-config
--libs --cflags,看看輸出結果

現在的問題就是怎樣用這些輸出結果了,最笨的方法就是複製貼上或者照抄,聰明的辦
法是在編譯命令列里加入這個`xxxx-config --libs --cflags`,比如編譯一個gtk程式:gcc gtktest.c `gtk-config --libs --cflags`這樣
就差不多了。注意`不是單引號,而是1鍵左邊那個鍵。

除了xxx-config以外,現在新的開發包一般都用pkg-config來生成連結引數,使用方法
跟xxx-config類似,但xxx-config是針對特定的開發包,但pkg-config包含很多開發包的連結引數的生成,用pkg- config --list-all命令可以列出所支援的所有開發包,pkg-config的用法就是pkg -config pagName --libs --cflags,其中pagName是包名,是pkg-config--list-all裡列出名單中的一個,比如gtk1.2的名字就是gtk+, pkg-

config gtk+ --libs --cflags的作用跟gtk-config --libs --cflags是一樣的。比如:
gcc gtktest.c `pkg-config gtk+ --libs --cflags`


5、-include和-I引數

-include用來包含標頭檔案,但一般情況下包含標頭檔案都在原始碼裡用#include xxxxxx實現,-include引數很少用。-I引數是用來指定標頭檔案目錄,/usr/include目錄一般是不用指定的,gcc知道去那裡找,但是如果標頭檔案不在/usr/include裡我們就要用-I引數指定了,比如標頭檔案放在/myinclude目錄裡,那編譯命令列就要加上- I/myinclude引數了,如果不加你會得到一個"xxxx.h: No such file or directory"的錯誤。-I引數可以用相對路徑,比如標頭檔案在當前目錄,可以用-I.來指定。上面我們提到的--cflags引數就是用來生成-I 引數的

6、-O引數

這是一個程式最佳化引數,一般用-O2就是,用來最佳化程式用的,比如gcc test.c -O2,最佳化得到的程式比沒最佳化的要小,執行速度可能也有所提高

7、-shared引數
編譯動態庫時要用到,比如gcc -shared test.c -o libtest.so

8、幾個相關的環境變數
PKG_CONFIG_PATH:用來指定pkg-config用到的pc檔案的路徑,預設是/usr/lib/pkgconf
ig,pc檔案是文字檔案,副檔名是.pc,裡面定義開發包的安裝路徑,Libs引數和Cflags引數等等。
CC:用來指定c編譯器
CXX:用來指定cxx編譯器
LIBS:跟上面的--libs作用差不多
CFLAGS:跟上面的--cflags作用差不多
CC,CXX,LIBS,CFLAGS手動編譯時一般用不上,在做configure時有時用到,一般情況
下不用管
環境變數設定方法:export ENV_NAME=xxxxxxxxxxxxxxxxx

9、關於交叉編譯

交叉編譯通俗地講就是在一種平臺上編譯出能執行在體系結構不同的另一種平臺上,比
如在我們地PC平臺(X86 CPU)上編譯出能執行在sparc

CPU平臺上的程式,編譯得到的程式在X86 CPU平臺上是不能執行的,必須放到sparc CPU平臺上才能執行。當然兩個平臺用的都是linux

這種方法在異平臺移植和嵌入式開發時用得非常普遍

相對與交叉編譯,我們平常做的編譯就叫本地編譯,也就是在當前平臺編譯,編譯得到
的程式也是在本地執行

用來編譯這種程式的編譯器就叫交叉編譯器,相對來說,用來做本地編譯的就叫本地編
譯器,一般用的都是gcc,但這種gcc跟本地的gcc編譯器

是不一樣的,需要在編譯gcc時用特定的configure引數才能得到支援交叉編譯的gcc

為了不跟本地編譯器混淆,交叉編譯器的名字一般都有字首,比如sparc-xxxx-linux-gn
u-gcc,sparc-xxxx-linux-gnu-g++ 等等

10、交叉編譯器的使用方法
使用方法跟本地的gcc差不多,但有一點特殊的是:必須用-L和-I引數指定編譯器用spar
c系統的庫和標頭檔案,不能用本地(X86)的庫(標頭檔案有時可以用本地的)

例子:
sparc-xxxx-linux-gnu-gcc test.c -L/path/to/sparcLib -I/path/to/sparcInclude

gcc與g++

Linux 中最重要的軟體開發工具是 GCC。GCC 是 GNU 的 C 和 C++ 編譯器。實際上,GCC 能夠編譯三種語言:C、C++ 和 Object C(C 語言的一種物件導向擴充套件)。利用 gcc 命令可同時編譯並連線 C 和 C++ 源程式。

GCC 可同時用來編譯 C 程式和 C++ 程式。一般來說,C 編譯器透過原始檔的字尾名來判斷是 C 程式還是 C++ 程式。在 Linux 中,C 原始檔的字尾名為 .c,而 C++ 原始檔的字尾名為 .C 或 .cpp。

gcc 命令只能編譯 C++ 原始檔,而不能自動和 C++ 程式使用的庫連線。因此,通常使用 g++ 命令來完成 C++ 程式的編譯和連線,該程式會自動呼叫 gcc 實現編譯。

選項 解釋
-ansi 只支援 ANSI 標準的 C 語法。這一選項將禁止 GNU C 的某些特色,
例如 asm 或 typeof 關鍵詞。
-c 只編譯並生成目標檔案。
-DMACRO 以字串“1”定義 MACRO 宏。
-DMACRO=DEFN 以字串“DEFN”定義 MACRO 宏。
-E 只執行 C 預編譯器。
-g 生成除錯資訊。GNU 偵錯程式可利用該資訊。
-IDIRECTORY 指定額外的標頭檔案搜尋路徑DIRECTORY。
-LDIRECTORY 指定額外的函式庫搜尋路徑DIRECTORY。
-lLIBRARY 連線時搜尋指定的函式庫LIBRARY。
-m486 針對 486 進行程式碼最佳化。
-o FILE 生成指定的輸出檔案。用在生成可執行檔案時。
-O0 不進行最佳化處理。
-O 或 -O1 最佳化生成程式碼。
-O2 進一步最佳化。
-O3 比 -O2 更進一步最佳化,包括 inline 函式。
-shared 生成共享目標檔案。通常用在建立共享庫時。
-static 禁止使用共享連線。
-UMACRO 取消對 MACRO 宏的定義。
-w 不生成任何警告資訊。
-Wall 生成所有警告資訊。


GNU 的偵錯程式稱為 gdb

由於軟體專案越來越大,也增加了軟體專案管理的難度。在開發組中,每個成員都要保留一個副本,在開發中非常容易引起衝突。CVS 就是為了解決這個問題的。
a、修改同步,防止一名開發人員的修改覆蓋其他人的成果。(check out、read only)

b、維護不同的版本。(按 version 查詢)

c、可查詢歷史記錄。防止 bug 的再引入。(diff)

CVS 為了解決這個問題,採用的方式是:

當開發人員對原始碼進行修改時,修改的內容被登記(check in)到了 CVS 倉庫(repository)中。倉庫中儲存了程式碼的主控副本,以及歷次修改的歷史資訊。它不儲存檔案的每個版本,而只是簡單的記錄發生在每個版本間的不同,節省磁碟空間。它能做到:

a、使開發人員的目錄和倉庫保持一致。可以把自己的修改提交(commit)給倉庫,讓倉庫更新自己。

b、允許程式碼派生。可以進行測試,如果失敗,可以消除所做的修改,維持原結果。

c、檢索任何一個版本。

來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/7591490/viewspace-1032401/,如需轉載,請註明出處,否則將追究法律責任。