【C++編譯】gcc的-l參數和-L參數

分類:IT技術 時間:2016-10-08

今天在編譯服務的時候,出現了一個錯誤:

/usr/bin/ld: cannot find -lxxx

於是查了一下,這個錯誤是因為鏈接程序ld在指定目錄裏找不到libxxx.so這個庫。

那麽,上面所說的“指定目錄”是哪些目錄,以及 -l的作用是什麽呢?

-l參數:用來指定程序要鏈接的庫,-l參數緊接著就是庫名。這裏的庫名並非真正的庫文件名。以庫名為math的庫為例,他的庫文件名是libmath.so或者libmath.a(linux下的庫文件都要以lib開頭,其中.so是動態庫,.a是靜態庫)。可見,把庫文件名的頭lib和尾.so去掉就是庫名了。

現在,如果我們自已要用到一個第三方提供的庫libtest.so,那麽我們只要把libtest.so拷貝到/usr/lib裏,編譯時加上-ltest參數,就能用上libtest.so庫了。

實際上,以下兩個命令是等價的:

gcc -o mytest mytest.c /usr/lib/libtest.so
gcc -o mytest mytest.c -ltest

放在/lib和/usr/lib和/usr/local/lib裏的庫直接用-l參數就能鏈接了(如果是標準C語言庫,我們可以不指定其庫路徑和庫名稱)。

如果庫文件沒放在這三個目錄裏,而是放在其他目錄裏,這時我們只用-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只是一個鏈接,比如libm.so鏈接到/lib/libm.so.6,/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程序所需的gtk鏈接參數,xxx-config除了–libs參數外還有一個參數是–cflags,用來生成頭文件包含目錄的,也就是-I參數。

要使用輸出的結果,我們可以復制粘貼或者照抄,不過更好的辦法是在編譯命令行裏加入:

`xxxx-config --libs --cflags`

比如編譯一個gtk程序:

gcc gtktest.c `gtk-config --libs --cflags`

註意`不是單引號,而是1鍵左邊那個鍵。

-include用來包含頭文件,但一般情況下包含頭文件都在源碼裏用#include xxxxxx實現,因此-include參數很少用。

-I參數是用來指定頭文件目錄,/usr/include目錄一般是不用指定的,gcc知道去那裏找,但是如果頭文件不在/usr/include裏我們就要用-I參數指定了,比如頭文件放在/myinclude目錄裏,那編譯命令行就要加上-I/myinclude參數了,如果不加,你會得到一個”xxxx.h: No such file or directory”的錯誤。-I參數可以用相對路徑,比如頭文件在當前目錄,可以用-I.來指定。

在了解以上內容之後,我們再回過頭來看看最開始遇到的問題:

/usr/bin/ld: cannot find -lxxx

發生這種錯誤的原因有以下三種情形:
1. 系統沒有安裝相對應的lib
2. 相對應的lib版本不對
3. lib(.so檔)的symbolic link 不正確,沒有連結到正確的函式庫文件(.so)

解決方法:
1. 先判斷在/usr/lib 下的相對應的函式庫文件(.so) 的symbolic link 是否正確,若不正確改成正確的連結目標即可解決問題。

  1. 若不是symbolic link 的問題引起,而是系統缺少相對應的lib安裝lib即可解決。

下面補充下如何生成庫文件:

假設有test_a.cpp test_b.cpp兩個源文件。

生成so文件的命令:

g++ test_a.cpp test_b.cpp -fPIC -shared -o libtest.so

生成.a文件的命令:

gcc -c test_a.cpp
gcc -c test_b.cpp
ar -r libtest.a test_a.o test_b.o

Tags: cannot 第三方 C語言 local Linux

文章來源:


ads
ads

相關文章
ads

相關文章

ad