【转】Eclipse+CDT+Gcc编译选项控制
原文地址:http://www.oschina.net/question/4873_19441
如果我們的程序調用動態鏈接庫,當在cdt中運行的時候,可以通過run-->enviroment-->添加LD_LIBRARY_PATH 環境變量來設置查找動態鏈接庫文件的路徑。但是在運行的時候,程序去那里找動態鏈接庫呢?
??????? 方法主要有兩種。一種,是設置系統的LD_LIBRARY_PATH 環境變量(在eclipse里設置對外部正式運行的程序無效)。設置的方法見附錄資料三
??????? 另外一種,是將動態鏈接庫文件的路徑編譯進二進制可執行文件。那么在eclipse如何進行編譯呢?如下圖
好了,這下我們的程序就可以脫離eclipse運行了(,找了好久啊)。注意,不要設置錯誤了,是g++linker的miscellaneous。
????? 第二種方法的缺點是,庫的地址就不能變動了(在游戲等應用程序中用這種方法教好)。第一種方法的優點是通過更改LD_LIBRARY_PATH的值,可以變動庫文件的地址。它的缺點當然就是還要改動環境變量的值了。
---------------------------------------------------------可愛的分割線----------------------------------------------------------
Linux 下動態庫使用小結
. 靜態庫和動態庫的基本概念
靜 態庫,是在可執行程序連接時就已經加入到執行碼中,在物理上成為執行程序的一部分;使用靜態庫編譯的程序運行時無需該庫文件支持,哪里都可以用,但是生成
的可執行文件較大。動態庫,是在可執行程序啟動時加載到執行程序中,可以被多個可執行程序共享使用。使用動態庫編譯生成的程序相對較小,但運行時需要庫文 件支持,如果機器里沒有這些庫文件就不能運行。
2.
如何使用動態庫
如何程序在連接時使用了共享庫,就必
須在運行的時候能夠找到共享庫的位置。linux的可執行程序在執行的時候默認是先搜索/lib和/usr/lib這兩個目錄,然后按照 /etc/ld.so.conf里面的配置搜索絕對路徑。同時,Linux也提供了環境變量LD_LIBRARY_PATH供用戶選擇使用,用戶可以通過 設定它來查找除默認路徑之外的其他路徑,如查找/work/lib路徑,你可以在/etc/rc.d/rc.local或其他系統啟動后即可執行到的腳本
添加如下語句:LD_LIBRARY_PATH =/work/lib:$(LD_LIBRARY_PATH)。并且LD_LIBRARY_PATH路徑優先于系統默認路徑之前查找(詳細參考《使用
LD_LIBRARY_PATH》)。
不過LD_LIBRARY_PATH的設定作用是全局的,過多的使用可能會影響到其他應用程序的運 行,所以多用在調試。(LD_LIBRARY_PATH的缺陷和使用準則,可以參考《Why
LD_LIBRARY_PATH is bad》 )。通常情況下推薦還是使用gcc的-R或-rpath選項來在編譯時就指定庫的查找路徑,并且該庫的路徑信息保存在可執行文件中,運行時它會直接到該路
徑查找庫,避免了使用LD_LIBRARY_PATH環境變量查找。
3.庫的鏈接時路徑和運行時路徑
現代連 接器在處理動態庫時將鏈接時路徑(Link-time path)和運行時路徑(Run-time path)分開,用戶可以通過-L指定連接時庫的路徑,通過-R(或-rpath)指定程序運行時庫的路徑,大大提高了庫應用的靈活性。比如我們做嵌入式
移植時#arm-linux-gcc $(CFLAGS) –o target –L/work/lib/zlib/
-llibz-1.2.3 (work/lib/zlib下是交叉編譯好的zlib庫),將target編譯好后我們只要把zlib庫拷貝到開發板的系統默認路徑下即可。或者通過- rpath(或-R )、LD_LIBRARY_PATH指定查找路徑。??
if [ -f ~/.bashrc ]; then
. ~/.bashrc
fi # User specific environment and startup programs PATH=$PATH:/sbin:/usr/sbin:/usr/local/sbin
LD_LIBRARY_PATH=$LD_LIBRARY_PATH:/usr/local/lib export PATH
export LD_LIBRARY_PATH
unset USERNAME 還是不行Top 3 樓cheeralen(嵌入式)回復于 2004-10-20 11:46:58 得分 0 bash不會從profile文件里讀取環境變量的
csh才會從profile文件里讀取變量
所以你可以修改你的bash為csh!!!Top 4 樓bekars(渦輪增壓:沒有解決不了的問題,因為根本就沒有問題)回復于 2004-10-20 11:57:37 得分 0 樓上的不太明白,能不能說多一些Top 5 樓winux0(隨緣不變)回復于 2004-10-20 11:58:16 得分 0 應該不會吧,不然你的export的那些PATH也沒有用哦Top 6 樓bekars(渦輪增壓:沒有解決不了的問題,因為根本就沒有問題)回復于 2004-10-20 12:35:29 得分 0 不知道,你可以幫我試試嗎?我這里就是不行 .bash_profile # Get the aliases and functions
if [ -f ~/.bashrc ]; then
. ~/.bashrc
fi # User specific environment and startup programs PATH=$PATH:/sbin:/usr/sbin:/usr/local/sbin
LD_LIBRARY_PATH=$LD_LIBRARY_PATH:/usr/local/lib export PATH
export LD_LIBRARY_PATH
unset USERNAME??
眾所周知,Linux動態庫的默認搜索路徑是/lib和/usr/lib。動態庫被創建后,一般都復制到這兩個目錄中。當程序執行時需要某動態庫, 并且該動態庫還未加載到內存中,則系統會自動到這兩個默認搜索路徑中去查找相應的動態庫文件,然后加載該文件到內存中,這樣程序就可以使用該動態庫中的函 數,以及該動態庫的其它資源了。在Linux 中,動態庫的搜索路徑除了默認的搜索路徑外,還可以通過以下三種方法來指定。
方法一:在配置文件/etc/ld.so.conf中指定動態庫搜索路徑。
可以通過編輯配置文件/etc/ld.so.conf來指定動態庫的搜索路徑,該文件中每行為一個動態庫搜索路徑。每次編輯完該文件后,都必須運行命令ldconfig使修改后的配置生效。我們通過例1來說明該方法。
例1:
我們通過以下命令用源程序pos_conf.c(見程序1)來創建動態庫 libpos.so,詳細創建過程請參考文[1]。
# gcc -c pos_conf.c
???????? # gcc -shared -fPCI -o libpos.so pos_conf.o
???????? #
#include <stdio.h>
???????? void pos()
???????? {
???????????????? printf("/root/test/conf/lib\n");
????????? }
????????? 程序1: pos_conf.c
接著通過以下命令編譯main.c(見程序2)生成目標程序pos。
# gcc -o pos main.c -L. -lpos
????????? #
void pos();
??????? int main()
??????? {
???????????? pos();
????????????? return 0;
???????? }
??????? 程序2: main.c
然后把庫文件移動到目錄/root/test/conf/lib中。
# mkdir -p /root/test/conf/lib
???????? # mv libpos.so /root/test/conf/lib
????????? #
最后編輯配置文件/etc/ld.so.conf,在該文件中追加一行"/root/test/conf/lib"。
運行程序pos試試。
# ./pos
???????? ./pos: error while loading shared libraries: libpos.so: cannot open shared object file: No such file or directory
????????? #
出錯了,系統未找到動態庫libpos.so。找找原因,原來在編輯完配置文件/etc/ld.so.conf后,沒有運行命令ldconfig,所以剛才的修改還未生效。我們運行ldconfig后再試試。
# ldconfig
???????? # ./pos
????????? /root/test/conf/lib
?????????? #
程序pos運行成功,并且打印出正確結果。
方法二:通過環境變量LD_LIBRARY_PATH指定動態庫搜索路徑。
通過設定環境變量LD_LIBRARY_PATH也可以指定動態庫搜索路徑。當通過該環境變量指定多個動態庫搜索路徑時,路徑之間用冒號":"分隔。下面通過例2來說明本方法。
例2:
我們通過以下命令用源程序pos_env.c(見程序3)來創建動態庫libpos.so。
# gcc -c pos_env.c
???????? # gcc -shared -fPCI -o libpos.so pos_env.o
????????? #
#include <stdio.h>
??????? void pos()
???????? {
?????????????? printf("/root/test/env/lib\n");
????????? }
??????? 程序3: pos_env.c
測試用的可執行文件pos可以使用例1中的得到的目標程序pos,不需要再次編譯。因為pos_conf.c中的函數pos和pos_env.c中的函數pos 函數原型一致,且動態庫名相同,這就好比修改動態庫pos后重新創建該庫一樣。這也是使用動態庫的優點之一。
然后把動態庫libpos.so移動到目錄/root/test/conf/lib中。
# mkdir -p /root/test/env/lib
???????? # mv libpos.so /root/test/env/lib
????????? #
我們可以使用export來設置該環境變量,在設置該環境變量后所有的命令中,該環境變量都有效。
例如:
# export LD_LIBRARY_PATH=/root/test/env/lib
????? ?? #
但本文為了舉例方便,使用另一種設置環境變量的方法,既在命令前加環境變量設置,該環境變量只對該命令有效,當該命令執行完成后,該環境變量就無效了。如下述命令:
# LD_LIBRARY_PATH=/root/test/env/lib ./pos
??????? /root/test/env/lib
???????? #
程序pos運行成功,并且打印的結果是"/root/test/env/lib",正是程序pos_env.c中的函數pos的運行結果。因此程序pos搜索到的動態庫是/root/test/env/lib/libpos.so。
方法三:在編譯目標代碼時指定該程序的動態庫搜索路徑。
還可以在編譯目標代碼時指定程序的動態庫搜索路徑。這是通過gcc 的參數"-Wl,-rpath,"指定(如例3所示)。當指定多個動態庫搜索路徑時,路徑之間用冒號":"分隔。
例3:
我們通過以下命令用源程序pos.c(見程序4)來創建動態庫libpos.so。
# gcc -c pos.c
???????? # gcc -shared -fPCI -o libpos.so pos.o
????????? #
#include <stdio.h>
??????? void pos()
???????? {
?????????????? printf("./\n");
???????? }
???????? 程序4: pos.c
因為我們需要在編譯目標代碼時指定可執行文件的動態庫搜索路徑,所以需要用gcc命令重新編譯源程序main.c(見程序2)來生成可執行文件pos。
# gcc -o pos main.c -L. -lpos -Wl,-rpath,./
???????? #
再運行程序pos試試。
# ./pos
???????? ./
????????? #
程序pos運行成功,輸出的結果正是pos.c中的函數pos的運行結果。因此程序pos搜索到的動態庫是./libpos.so。
以上介紹了三種指定動態庫搜索路徑的方法,加上默認的動態庫搜索路徑/lib和/usr/lib,共五種動態庫的搜索路徑,那么它們搜索的先后順序是什么呢?
在介紹上述三種方法時,分別創建了動態庫./libpos.so、 /root/test/env/lib/libpos.so和/root/test/conf/lib/libpos.so。我們再用源程序 pos_lib.c(見程序5)來創建動態庫/lib/libpos.so,用源程序pos_usrlib.c(見程序6)來創建動態庫 /usr/lib/libpos.so。
#include <stdio.h>
???????? void pos()
????????? {
???????????????? printf("/lib\n");
?????????? }
????????? 程序5: pos_lib.c
#include <stdio.h>
??????? void pos()
???????? {
??????????????? printf("/usr/lib\n");
???????? }
??????? 程序6: pos_usrlib.c
這樣我們得到五個動態庫libpos.so,這些動態庫的名字相同,且都包含相同函數原型 的公用函數pos。但存儲的位置不同和公用函數pos 打印的結果不同。每個動態庫中的公用函數pos都輸出該動態庫所存放的位置。這樣我們可以通過執行例3中的可執行文件pos得到的結果不同獲知其搜索到了 哪個動態庫,從而獲得第1個動態庫搜索順序,然后刪除該動態庫,再執行程序pos,獲得第2個動態庫搜索路徑,再刪除第2個被搜索到的動態庫,如此往復, 將可得到Linux搜索動態庫的先后順序。程序pos執行的輸出結果和搜索到的動態庫的對應關系如表1所示:
| ./ | ./libpos.so | 編譯目標代碼時指定的動態庫搜索路徑 |
| /root/test/env/lib | /root/test/env/lib/libpos.so | 環境變量LD_LIBRARY_PATH指定的動態庫搜索路徑 |
| /root/test/conf/lib | /root/test/conf/lib/libpos.so | 配置文件/etc/ld.so.conf中指定的動態庫搜索路徑 |
| /lib | /lib/libpos.so | 默認的動態庫搜索路徑/lib |
| /usr/lib | /usr/lib/libpos.so | 默認的動態庫搜索路徑/usr/lib |
創建各個動態庫,并放置在相應的目錄中。測試環境就準備好了。執行程序pos,并在該命令行中設置環境變量LD_LIBRARY_PATH。
# LD_LIBRARY_PATH=/root/test/env/lib ./pos
????????? ./
????????? #
根據程序pos的輸出結果可知,最先搜索的是編譯目標代碼時指定的動態庫搜索路徑。然后我們把動態庫./libpos.so刪除了,再運行上述命令試試。
# rm libpos.so
??????? rm: remove regular file `libpos.so'? y
??????? # LD_LIBRARY_PATH=/root/test/env/lib ./pos
??????? /root/test/env/lib
??????? #
根據程序pos的輸出結果可知,第2個動態庫搜索的路徑是環境變量LD_LIBRARY_PATH指定的。我們再把/root/test/env/lib/libpos.so刪除,運行上述命令。
# rm /root/test/env/lib/libpos.so
???????? rm: remove regular file `/root/test/env/lib/libpos.so'? y
??????? # LD_LIBRARY_PATH=/root/test/env/lib ./pos
??????? /root/test/conf/lib
??????? #
第3個動態庫的搜索路徑是配置文件/etc/ld.so.conf指定的路徑。刪除動態庫/root/test/conf/lib/libpos.so后再運行上述命令。
# rm /root/test/conf/lib/libpos.so
??????? rm: remove regular file `/root/test/conf/lib/libpos.so'? y
??????? # LD_LIBRARY_PATH=/root/test/env/lib ./pos
??????? /lib
??????? #
第4個動態庫的搜索路徑是默認搜索路徑/lib。我們再刪除動態庫/lib/libpos.so,運行上述命令。
# rm /lib/libpos.so
??????? rm: remove regular file `/lib/libpos.so'? y
??????? # LD_LIBRARY_PATH=/root/test/env/lib ./pos
??????? /usr/lib
??????? #
最后的動態庫搜索路徑是默認搜索路徑/usr/lib。
綜合以上結果可知,動態庫的搜索路徑搜索的先后順序是:
1.編譯目標代碼時指定的動態庫搜索路徑;
2.環境變量LD_LIBRARY_PATH指定的動態庫搜索路徑;
3.配置文件/etc/ld.so.conf中指定的動態庫搜索路徑;
4.默認的動態庫搜索路徑/lib;
5.默認的動態庫搜索路徑/usr/lib。
在上述1、2、3指定動態庫搜索路徑時,都可指定多個動態庫搜索路徑,其搜索的先后順序是按指定路徑的先后順序搜索的。對此本文不再舉例說明,有興趣的讀者可以參照本文的方法驗證。
轉載于:https://www.cnblogs.com/ainubis/p/3985876.html
總結
以上是生活随笔為你收集整理的【转】Eclipse+CDT+Gcc编译选项控制的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: pdftk的使用介绍
- 下一篇: MVC.Net: jqueryval错误