看到乙個兄弟的部落格,其中對g++後面引數-l的學習非常好,收藏起來,以備後面學習。
為什麼會出現undefined reference to '***xx'錯誤?
首先這是鏈結錯誤,不是編譯錯誤,也就是說如果只有這個錯誤,說明你的程式原始碼本身沒有問題,是你用編譯器編譯時引數用得不對,你沒有指定鏈結程式要用到得庫,比如你的程式裡用到了一些數學函式,那麼你就要在編譯引數裡指定程式要鏈結數學庫,方法是在編譯命令列裡加入-lm。
-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-l***」,也就是鏈結程式ld在那3個目錄裡找不到lib***.so,這時另外乙個引數-l就派上用場了。
-l引數跟著的是庫檔案所在的目錄名,比如我們把libtest.so放在/aaa/bbb/ccc目錄下,那鏈結引數就是-l/aaa/bbb/ccc -ltest 。
另外,大部分lib***x.so只是乙個鏈結,以rh9為例,比如libm.so它鏈結到/lib/libm.so.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程式所需的gtk鏈結引數。
***-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`。
-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引數的。
-o引數
這是乙個程式優化引數,一般用-o2就是,用來優化程式用的,比如gcc test.c -o2,優化得到的程式比沒優化的要小,執行速度可能也有所提高。
-shared引數
編譯動態庫時要用到,比如gcc -shared test.c -o libtest.so
幾個相關的環境變數
pkg_config_path:用來指定pkg-config用到的pc檔案的路徑,預設是/usr/lib/pkgconfig,pc檔案是文字檔案,副檔名是.pc,裡面定義開發包的安裝路徑,libs引數和cflags引數等等。
cc:用來指定c編譯器。
cxx:用來指定cxx編譯器。
libs:跟上面的--libs作用差不多。
cflags:跟上面的--cflags作用差不多。
cc,cxx,libs,cflags手動編譯時一般用不上,在做configure時有時用到,一般情況下不用管。
環境變數設定方法:export env_name=***************xx
生成靜態庫及其使用
ar cr name.a *.o
g++ -o runname *.o path/to/name.a
注:name.a 要放在*.o後面,而不是與-lm等放一起。
生成動態庫及其使用
g++ -shared -o libname.so *.o
g++ -wl,-bsymbolic -fpic -shared -o libname.a *.o
-wl,-bsymbolic 表示繫結自己的symbol。
-fpic 使輸出的物件模組是按照可重定位位址方式生成的。
-shared 指定把對應的原始檔生成對應的動態鏈結庫檔案libname.so檔案。
g++ -o runname *.o -lname -lpath/to
gcc g 命令引數筆記
1.gcc e source file.c e,只執行到預編譯。直接輸出預編譯結果。2.gcc s source file.c s,只執行到源 到彙編 的轉換,輸出彙編 3.gcc c source file.c c,只執行到編譯,輸出目標檔案。4.gcc e s c source file.c o...
GCC G 基本用法
一.常用編譯命令選項 假設源程式檔名為test.c。1.無選項編譯鏈結 用法 gcc test.c 作用 將test.c預處理 彙編 編譯並鏈結形成可執行檔案。這裡未指定輸出檔案,預設輸出為a.out。編譯成功後可以看到生成了乙個a.out的檔案。在命令列輸入.a.out 執行程式。表示在當前目錄,...
gcc g 常用引數
注 以下1.c和1.cpp均為原始檔 1.gcc 1.c和g 1.cpp 為直接生成乙個a.exe的檔案 2.gcc o 1.exe 1.c和g o 1.exe 1.cpp 為生成乙個1.exe的檔案 3.gcc s 1.c和g s 1.cpp 只啟用預處理和編譯,就是指把檔案編譯成為彙編 4.gc...