Linux動(dòng)態(tài)庫(.so)搜索路徑
眾所周知,Linux動(dòng)態(tài)庫的默認(rèn)搜索路徑是/lib和/usr/lib。動(dòng)態(tài)庫被創(chuàng)建后,一般都復(fù)制到這兩個(gè)目錄中。當(dāng)程序執(zhí)行時(shí)需要某動(dòng)態(tài)庫, 并且該 動(dòng) 態(tài)庫還未加載到內(nèi)存中,則系統(tǒng)會(huì)自動(dòng)到這兩個(gè)默認(rèn)搜索路徑中去查找相應(yīng)的動(dòng)態(tài)庫文件,然后加載該文件到內(nèi)存中,這樣程序就可以使用該動(dòng)態(tài)庫中的函數(shù),以及 該動(dòng)態(tài)庫的其它資源了。在Linux 中,動(dòng)態(tài)庫的搜索路徑除了默認(rèn)的搜索路徑外,還可以通過以下三種方法來指定。
方法一:在配置文件/etc/ld.so.conf中指定動(dòng)態(tài)庫搜索路徑。
可以通過編輯配置文件/etc/ld.so.conf來指定動(dòng)態(tài)庫的搜索路徑,該文件中每行為一個(gè)動(dòng)態(tài)庫搜索路徑。每次編輯完該文件后,都必須運(yùn)行命令ldconfig使修改后的配置生效。我們通過例1來說明該方法。
例1:
我們通過以下命令用源程序pos_conf.c(見程序1)來創(chuàng)建動(dòng)態(tài)庫 libpos.so,詳細(xì)創(chuàng)建過程請(qǐng)參考文[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)生成目標(biāo)程序pos。
# gcc -o pos main.c -L. -lpos
#
void pos();
int main()
{
pos();
return 0;
}
程序2: main.c
然后把庫文件移動(dòng)到目錄/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"。
運(yùn)行程序pos試試。
# ./pos
./pos: error while loading shared libraries: libpos.so: cannot open shared object file: No such file or directory
#
出錯(cuò)了,系統(tǒng)未找到動(dòng)態(tài)庫libpos.so。找找原因,原來在編輯完配置文件/etc/ld.so.conf后,沒有運(yùn)行命令ldconfig,所以剛才的修改還未生效。我們運(yùn)行l(wèi)dconfig后再試試。
# ldconfig
# ./pos
/root/test/conf/lib
#
程序pos運(yùn)行成功,并且打印出正確結(jié)果。
方法二:通過環(huán)境變量LD_LIBRARY_PATH指定動(dòng)態(tài)庫搜索路徑。
通過設(shè)定環(huán)境變量LD_LIBRARY_PATH也可以指定動(dòng)態(tài)庫搜索路徑。當(dāng)通過該環(huán)境變量指定多個(gè)動(dòng)態(tài)庫搜索路徑時(shí),路徑之間用冒號(hào)":"分隔。下面通過例2來說明本方法。
例2:
我們通過以下命令用源程序pos_env.c(見程序3)來創(chuàng)建動(dòng)態(tài)庫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
測(cè)試用的可執(zhí)行文件pos可以使用例1中的得到的目標(biāo)程序pos,不需要再次編譯。因?yàn)閜os_conf.c中的函數(shù)pos和pos_env.c中的函數(shù)pos 函數(shù)原型一致,且動(dòng)態(tài)庫名相同,這就好比修改動(dòng)態(tài)庫pos后重新創(chuàng)建該庫一樣。這也是使用動(dòng)態(tài)庫的優(yōu)點(diǎn)之一。
然后把動(dòng)態(tài)庫libpos.so移動(dòng)到目錄/root/test/conf/lib中。
# mkdir -p /root/test/env/lib
# mv libpos.so /root/test/env/lib
#
我們可以使用export來設(shè)置該環(huán)境變量,在設(shè)置該環(huán)境變量后所有的命令中,該環(huán)境變量都有效。
例如:
# export LD_LIBRARY_PATH=/root/test/env/lib
#
但本文為了舉例方便,使用另一種設(shè)置環(huán)境變量的方法,既在命令前加環(huán)境變量設(shè)置,該環(huán)境變量只對(duì)該命令有效,當(dāng)該命令執(zhí)行完成后,該環(huán)境變量就無效了。如下述命令:
# LD_LIBRARY_PATH=/root/test/env/lib ./pos
/root/test/env/lib
#
程序pos運(yùn)行成功,并且打印的結(jié)果是"/root/test/env/lib",正是程序pos_env.c中的函數(shù)pos的運(yùn)行結(jié)果。因此程序pos搜索到的動(dòng)態(tài)庫是/root/test/env/lib/libpos.so。
方法三:在編譯目標(biāo)代碼時(shí)指定該程序的動(dòng)態(tài)庫搜索路徑。
還可以在編譯目標(biāo)代碼時(shí)指定程序的動(dòng)態(tài)庫搜索路徑。這是通過gcc 的參數(shù)"-Wl,-rpath,"指定(如例3所示)。當(dāng)指定多個(gè)動(dòng)態(tài)庫搜索路徑時(shí),路徑之間用冒號(hào)":"分隔。
例3:
我們通過以下命令用源程序pos.c(見程序4)來創(chuàng)建動(dòng)態(tài)庫libpos.so。
# gcc -c pos.c
# gcc -shared -fPCI -o libpos.so pos.o
#
#include <stdio.h>
void pos()
{
printf("./\n");
}
程序4: pos.c
因?yàn)槲覀冃枰诰幾g目標(biāo)代碼時(shí)指定可執(zhí)行文件的動(dòng)態(tài)庫搜索路徑,所以需要用gcc命令重新編譯源程序main.c(見程序2)來生成可執(zhí)行文件pos。
# gcc -o pos main.c -L. -lpos -Wl,-rpath,./
#
再運(yùn)行程序pos試試。
# ./pos
./
#
程序pos運(yùn)行成功,輸出的結(jié)果正是pos.c中的函數(shù)pos的運(yùn)行結(jié)果。因此程序pos搜索到的動(dòng)態(tài)庫是./libpos.so。
以上介紹了三種指定動(dòng)態(tài)庫搜索路徑的方法,加上默認(rèn)的動(dòng)態(tài)庫搜索路徑/lib和/usr/lib,共五種動(dòng)態(tài)庫的搜索路徑,那么它們搜索的先后順序是什么呢?
在 介紹上述三種方法時(shí),分別創(chuàng)建了動(dòng)態(tài)庫./libpos.so、 /root/test/env/lib/libpos.so和/root/test/conf/lib/libpos.so。我們?cè)儆迷闯绦? pos_lib.c(見程序5)來創(chuàng)建動(dòng)態(tài)庫/lib/libpos.so,用源程序pos_usrlib.c(見程序6)來創(chuàng)建動(dòng)態(tài)庫 /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
這 樣我們得到五個(gè)動(dòng)態(tài)庫libpos.so,這些動(dòng)態(tài)庫的名字相同,且都包含相同函數(shù)原型的公用函數(shù)pos。但存儲(chǔ)的位置不同和公用函數(shù)pos 打印的結(jié)果不同。每個(gè)動(dòng)態(tài)庫中的公用函數(shù)pos都輸出該動(dòng)態(tài)庫所存放的位置。這樣我們可以通過執(zhí)行例3中的可執(zhí)行文件pos得到的結(jié)果不同獲知其搜索到了 哪個(gè)動(dòng)態(tài)庫,從而獲得第1個(gè)動(dòng)態(tài)庫搜索順序,然后刪除該動(dòng)態(tài)庫,再執(zhí)行程序pos,獲得第2個(gè)動(dòng)態(tài)庫搜索路徑,再刪除第2個(gè)被搜索到的動(dòng)態(tài)庫,如此往復(fù), 將可得到Linux搜索動(dòng)態(tài)庫的先后順序。程序pos執(zhí)行的輸出結(jié)果和搜索到的動(dòng)態(tài)庫的對(duì)應(yīng)關(guān)系如表1所示:
程序pos輸出結(jié)果
使用的動(dòng)態(tài)庫
對(duì)應(yīng)的動(dòng)態(tài)庫搜索路徑指定方式
./
./libpos.so
編譯目標(biāo)代碼時(shí)指定的動(dòng)態(tài)庫搜索路徑
/root/test/env/lib
/root/test/env/lib/libpos.so
環(huán)境變量LD_LIBRARY_PATH指定的動(dòng)態(tài)庫搜索路徑
/root/test/conf/lib
/root/test/conf/lib/libpos.so
配置文件/etc/ld.so.conf中指定的動(dòng)態(tài)庫搜索路徑
/lib
/lib/libpos.so
默認(rèn)的動(dòng)態(tài)庫搜索路徑/lib
/usr/lib
/usr/lib/libpos.so
默認(rèn)的動(dòng)態(tài)庫搜索路徑/usr/lib
表1: 程序pos輸出結(jié)果和動(dòng)態(tài)庫的對(duì)應(yīng)關(guān)系
創(chuàng)建各個(gè)動(dòng)態(tài)庫,并放置在相應(yīng)的目錄中。測(cè)試環(huán)境就準(zhǔn)備好了。執(zhí)行程序pos,并在該命令行中設(shè)置環(huán)境變量LD_LIBRARY_PATH。
# LD_LIBRARY_PATH=/root/test/env/lib ./pos
./
#
根據(jù)程序pos的輸出結(jié)果可知,最先搜索的是編譯目標(biāo)代碼時(shí)指定的動(dòng)態(tài)庫搜索路徑。然后我們把動(dòng)態(tài)庫./libpos.so刪除了,再運(yùn)行上述命令試試。
# rm libpos.so
rm: remove regular file `libpos.so'? y
# LD_LIBRARY_PATH=/root/test/env/lib ./pos
/root/test/env/lib
#
根據(jù)程序pos的輸出結(jié)果可知,第2個(gè)動(dòng)態(tài)庫搜索的路徑是環(huán)境變量LD_LIBRARY_PATH指定的。我們?cè)侔?root/test/env/lib/libpos.so刪除,運(yùn)行上述命令。
# 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個(gè)動(dòng)態(tài)庫的搜索路徑是配置文件/etc/ld.so.conf指定的路徑。刪除動(dòng)態(tài)庫/root/test/conf/lib/libpos.so后再運(yùn)行上述命令。
# 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個(gè)動(dòng)態(tài)庫的搜索路徑是默認(rèn)搜索路徑/lib。我們?cè)賱h除動(dòng)態(tài)庫/lib/libpos.so,運(yùn)行上述命令。
# rm /lib/libpos.so
rm: remove regular file `/lib/libpos.so'? y
# LD_LIBRARY_PATH=/root/test/env/lib ./pos
/usr/lib
#
最后的動(dòng)態(tài)庫搜索路徑是默認(rèn)搜索路徑/usr/lib。
綜合以上結(jié)果可知,動(dòng)態(tài)庫的搜索路徑搜索的先后順序是:
1.編譯目標(biāo)代碼時(shí)指定的動(dòng)態(tài)庫搜索路徑;
2.環(huán)境變量LD_LIBRARY_PATH指定的動(dòng)態(tài)庫搜索路徑;
3.配置文件/etc/ld.so.conf中指定的動(dòng)態(tài)庫搜索路徑;
4.默認(rèn)的動(dòng)態(tài)庫搜索路徑/lib;
5.默認(rèn)的動(dòng)態(tài)庫搜索路徑/usr/lib。
在上述1、2、3指定動(dòng)態(tài)庫搜索路徑時(shí),都可指定多個(gè)動(dòng)態(tài)庫搜索路徑,其搜索的先后順序是按指定路徑的先后順序搜索的。對(duì)此本文不再舉例說明,有興趣的讀者可以參照本文的方法驗(yàn)證。
posted on 2012-11-06 18:51 大龍 閱讀(242) 評(píng)論(0) 編輯 收藏 引用