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 '***xx'錯誤?
首先這是鏈結錯誤,不是編譯錯誤,也就是說如果只有這個錯誤,說明你的程式原始碼本身沒有問題,是你用編譯器編譯時引數用得不對,你沒有指定鏈結程式要用到得庫,比如你的程式裡用到了一些數學函式,那麼你就要在編譯引數裡指定程式要鏈結數學庫,方法是在編譯命令列裡加入-lm
4、-l引數和-l引數
-l引數就是用來指定程式要鏈結的庫,-l引數緊接著就是庫名,那麼庫名跟真正的庫檔名有什麼關係呢?-lname,在連線時,裝載名字為「libname.a」的函式庫:-lm表示連線名為「libm.a」的數學函式庫。就拿數學庫來說,他的庫名是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 -l***」,也就是鏈結程式ld在那3個目錄裡找不到lib***.so,這時另外乙個引數-l就派上用場了,比如常用的x11的庫,它在/usr/x11r6/lib目錄下,我們編譯時就要用-l/usr/x11r6/lib -lx11引數,-l引數跟著的是庫檔案所在的目錄名。再比如我們把libtest.so放在/aaa/bbb/ccc目錄下,那鏈結引數就是-l/aaa/bbb/ccc -ltest
另外,大部分lib***x.so只是乙個鏈結,以rh9為例,比如libm.so它鏈結到/lib/libm.s
o.x,/lib/libm.so.6又鏈結到/lib/libm-2.3.2.so,
如果沒有這樣的鏈結,還是會出錯,因為ld只會找lib***x.so,所以如果你要用到***x
庫,而只有lib***x.so.x或者lib***x-x.x.x.so,做乙個鏈結就可以了
ln -s lib***x-x.x.x.so lib***x.so
手工來寫鏈結引數總是很麻煩的,還好很多庫開發包提供了生成鏈結引數的程式,名字一般叫***x-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鏈結引數,***-config除了--libs引數外還有乙個引數是--cflags用來生成頭文
件包含目錄的,也就是-i引數,在下面我們將會講到。你可以試試執行gtk-config
--libs --cflags,看看輸出結果
現在的問題就是怎樣用這些輸出結果了,最笨的方法就是複製貼上或者照抄,聰明的辦
法是在編譯命令列裡加入這個`***x-config --libs --cflags`,比如編譯乙個gtk程式:gcc gtktest.c `gtk-config --libs --cflags`這樣
就差不多了。注意`不是單引號,而是1鍵左邊那個鍵。
除了***-config以外,現在新的開發包一般都用pkg-config來生成鏈結引數,使用方法
跟***-config類似,但***-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 ******實現,-include引數很少用。-i引數是用來指定頭檔案目錄,/usr/include目錄一般是不用指定的,gcc知道去那裡找,但是如果標頭檔案不在/usr/include裡我們就要用-i引數指定了,比如標頭檔案放在/myinclude目錄裡,那編譯命令列就要加上-i/myinclude引數了,如果不加你會得到乙個"***x.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=***************xx
9、關於交叉編譯
交叉編譯通俗地講就是在一種平台上編譯出能執行在體系結構不同的另一種平台上,比
如在我們地pc平台(x86 cpu)上編譯出能執行在sparc
cpu平台上的程式,編譯得到的程式在x86 cpu平台上是不能執行的,必須放到sparc cpu平台上才能執行。當然兩個平台用的都是linux
這種方法在異平台移植和嵌入式開發時用得非常普遍
相對與交叉編譯,我們平常做的編譯就叫本地編譯,也就是在當前平台編譯,編譯得到
的程式也是在本地執行
用來編譯這種程式的編譯器就叫交叉編譯器,相對來說,用來做本地編譯的就叫本地編
譯器,一般用的都是gcc,但這種gcc跟本地的gcc編譯器
是不一樣的,需要在編譯gcc時用特定的configure引數才能得到支援交叉編譯的gcc
為了不跟本地編譯器混淆,交叉編譯器的名字一般都有字首,比如sparc-***x-linux-gn
u-gcc,sparc-***x-linux-gnu-g++ 等等
10、交叉編譯器的使用方法
使用方法跟本地的gcc差不多,但有一點特殊的是:必須用-l和-i引數指定編譯器用spar
c系統的庫和標頭檔案,不能用本地(x86)的庫(標頭檔案有時可以用本地的)
例子:sparc-***x-linux-gnu-gcc test.c -l/path/to/sparclib -i/path/to/sparcinclude
gcc與g++
gcc編譯opengl: gcc hello.c -o hello -l/usr/x11r6/lib/ -lgl -lglu -lglut
**:
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這樣將...
Gcc命令列詳解
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這樣將編譯出乙個名為te...
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這樣將...