ldd
創(chuàng)新互聯(lián)公司基于成都重慶香港及美國等地區(qū)分布式IDC機房數(shù)據(jù)中心構(gòu)建的電信大帶寬,聯(lián)通大帶寬,移動大帶寬,多線BGP大帶寬租用,是為眾多客戶提供專業(yè)雅安機房托管報價,主機托管價格性價比高,為金融證券行業(yè)服務(wù)器托管,ai人工智能服務(wù)器托管提供bgp線路100M獨享,G口帶寬及機柜租用的專業(yè)成都idc公司。
可執(zhí)行文件名
查看可執(zhí)行文件鏈接了哪些
系統(tǒng)動態(tài)鏈接庫
nm
可執(zhí)行文件名
查看可執(zhí)行文件里面有哪些符號
strip
可執(zhí)行文件名
去除符號表可以給可執(zhí)行文件瘦身
如果我們想從可執(zhí)行程序里面提取出來一點什么文本信息的話,還可以用strings命令
strings
可執(zhí)行文件名
Linux操作系統(tǒng)上面的動態(tài)共享庫大致分為三類:
1、操作系統(tǒng)級別的共享庫和基礎(chǔ)的系統(tǒng)工具庫
比方說libc.so,
libz.so,
libpthread.so等等,這些系統(tǒng)庫會被放在/lib和/usr/lib目錄下面,如果是64位操作系統(tǒng),還會有/lib64和/usr
/lib64目錄。如果操作系統(tǒng)帶有圖形界面,那么還會有/usr/X11R6/lib目錄,如果是64位操作系統(tǒng),還有/usr/X11R6
/lib64目錄。此外還可能有其他特定Linux版本的系統(tǒng)庫目錄。
這些系統(tǒng)庫文件的完整和版本的正確,確保了Linux上面各種程序能夠正常的運行。
2、應(yīng)用程序級別的系統(tǒng)共享庫
并非操作系統(tǒng)自帶,但是可能被很多應(yīng)用程序所共享的庫,一般會被放在/usr/local/lib和/usr/local/lib64這兩個目錄下面。很多你自行編譯安裝的程序都會在編譯的時候自動把/usr/local/lib加入gcc的-L參數(shù),而在運行的時候自動到/usr/local
/lib下面去尋找共享庫。
以上兩類的動態(tài)共享庫,應(yīng)用程序會自動尋找到他們,并不需要你額外的設(shè)置和擔(dān)心。這是為什么呢?因為以上這些目錄默認就被加入到動態(tài)鏈接程序的搜索路徑里面了。Linux的系統(tǒng)共享庫搜索路徑定義在/etc/ld.so.conf這個配置文件里面。這個文件的內(nèi)容格式大致如下:
/usr/X11R6/lib64
/usr/X11R6/lib
/usr/local/lib
/lib64
/lib
/usr/lib64
/usr/lib
/usr/local/lib64
/usr/local/ImageMagick/lib
假設(shè)我們自己編譯安裝的ImageMagick圖形庫在/usr/local/ImageMagick目錄下面,并且希望其他應(yīng)用程序都可以使用
ImageMagick的動態(tài)共享庫,那么我們只需要把/usr/local/ImageMagick/lib目錄加入/etc/ld.so.conf文件里面,然后執(zhí)行:ldconfig
命令即可。
ldcofig將搜索以上所有的目錄,為共享庫建立一個緩存文件/etc/ld.so.cache。為了確認ldconfig已經(jīng)搜索到ImageMagick的庫,我們可以用上面介紹的strings命令從ld.so.cache里面抽取文本信息來檢查一下:
strings
/etc/ld.so.cache
|
grep
ImageMagick
輸出結(jié)果為:
/usr/local/ImageMagick/lib/libWand.so.10
/usr/local/ImageMagick/lib/libWand.so
/usr/local/ImageMagick/lib/libMagick.so.10
/usr/local/ImageMagick/lib/libMagick.so
/usr/local/ImageMagick/lib/libMagick++.so.10
/usr/local/ImageMagick/lib/libMagick++.so
已經(jīng)成功了!
3、應(yīng)用程序獨享的動態(tài)共享庫
有很多共享庫只被特定的應(yīng)用程序使用,那么就沒有必要加入系統(tǒng)庫路徑,以免應(yīng)用程序的共享庫之間發(fā)生版本沖突。因此Linux還可以通過設(shè)置環(huán)境變量LD_LIBRARY_PATH來臨時指定應(yīng)用程序的共享庫搜索路徑,就像我們上面舉的那個例子一樣,我們可以在應(yīng)用程序的啟動腳本里面預(yù)先設(shè)置
LD_LIBRARY_PATH,指定本應(yīng)用程序附加的共享庫搜索路徑,從而讓應(yīng)用程序找到它。
用JNI實現(xiàn)
實例:
創(chuàng)建HelloWorld.java
class HelloWorld
{
private native void print();
public staticvoid main(String[] args)
{
new HelloWorld().print();
}
static
{
System.loadLibrary("HelloWorld");
}
}
注意print方法的聲明,關(guān)鍵字native表明該方法是一個原生代碼實現(xiàn)的。另外注意static代碼段的System.loadLibrary調(diào)用,這段代碼表示在程序加載的時候,自動加載libHelloWorld.so庫。
編譯HelloWorld.java
在命令行中運行如下命令:
javac HelloWorld.java
在當(dāng)前文件夾編譯生成HelloWorld.class。
生成HelloWorld.h
在命令行中運行如下命令:
javah -jni HelloWorld
在當(dāng)前文件夾中會生成HelloWorld.h。打開HelloWorld.h將會發(fā)現(xiàn)如下代碼:
/* DO NOT EDIT THIS FILE - it is machine generated */
#include jni.h
/* Header for class HelloWorld */
#ifndef _Included_HelloWorld
#define _Included_HelloWorld
#ifdef __cplusplus
extern "C" {
#endif
/*
* Class: HelloWorld
* Method: print
* Signature: ()V
*/
JNIEXPORT void JNICALL Java_HelloWorld_print
(JNIEnv *, jobject);
#ifdef __cplusplus
}
#endif
#endif
該文件中包含了一個函數(shù)Java_HelloWorld_print的聲明。這里面包含兩個參數(shù),非常重要,后面講實現(xiàn)的時候會講到。
實現(xiàn)HelloWorld.c
創(chuàng)建HelloWorld.c文件輸入如下的代碼:
#include jni.h
#include stdio.h
#include "HelloWorld.h"
JNIEXPORT void JNICALL
Java_HelloWorld_print(JNIEnv *env, jobject obj)
{
printf("Hello World!\n");
}
注意必須要包含jni.h頭文件,該文件中定義了JNI用到的各種類型,宏定義等。
另外需要注意Java_HelloWorld_print的兩個參數(shù),本例比較簡單,不需要用到這兩個參數(shù)。但是這兩個參數(shù)在JNI中非常重要。
env代表java虛擬機環(huán)境,Java傳過來的參數(shù)和c有很大的不同,需要調(diào)用JVM提供的接口來轉(zhuǎn)換成C類型的,就是通過調(diào)用env方法來完成轉(zhuǎn)換的。
obj代表調(diào)用的對象,相當(dāng)于c++的this。當(dāng)c函數(shù)需要改變調(diào)用對象成員變量時,可以通過操作這個對象來完成。
編譯生成libHelloWorld.so
在Linux下執(zhí)行如下命令來完成編譯工作:
cc -I/usr/lib/jvm/java-6-sun/include/linux/
-I/usr/lib/jvm/java-6-sun/include/
-fPIC -shared -o libHelloWorld.so HelloWorld.c
在當(dāng)前目錄生成libHelloWorld.so。注意一定需要包含Java的include目錄(請根據(jù)自己系統(tǒng)環(huán)境設(shè)定),因為Helloworld.c中包含了jni.h。
另外一個值得注意的是在HelloWorld.java中我們LoadLibrary方法加載的是
“HelloWorld”,可我們生成的Library卻是libHelloWorld。這是Linux的鏈接規(guī)定的,一個庫的必須要是:lib+庫
名+.so。鏈接的時候只需要提供庫名就可以了。
運行Java程序HelloWorld
大功告成最后一步,驗證前面的成果的時刻到了:
java HelloWorld
如果你這步發(fā)生問題,如果這步你收到j(luò)ava.lang.UnsatisfiedLinkError異常,可以通過如下方式指明共享庫的路徑:
java -Djava.library.path='.' HelloWorld
當(dāng)然還有其他的方式可以指明路徑請參考《在Linux平臺下使用JNI》。
我們可以看到久違的“Hello world!”輸出了。
新建一個sort.c文件,寫一個最簡單的排序
使用 gcc -o libsort.so -fPIC -shared sort.c 產(chǎn)生libsort.so庫。
.so庫有兩種調(diào)用方法:
新建main.c文件:
使用命令 gcc -o main main.c -lsort -L. 編譯。
新建main2.c文件:
使用命令 gcc -o main2 main2.c -ldl 編譯。動態(tài)加載.so庫的話需要-ldl。
運行./main2后輸出遞增序列,調(diào)用成功。
1、 .so動態(tài)庫的生成
可使用gcc或者g++編譯器生成動態(tài)庫文件(此處以g++編譯器為例)
g++ -shared -fPIC -c XXX.cpp
g++ -shared -fPIC -o XXX.so XXX.o
2、 .so動態(tài)庫的動態(tài)調(diào)用接口函數(shù)說明
動態(tài)庫的調(diào)用關(guān)系可以在需要調(diào)用動態(tài)庫的程序編譯時,通過g++的-L和-l命令來指定。例如:程序test啟動時需要加載目錄/root/src/lib中的libtest_so1.so動態(tài)庫,編譯命令可照如下編寫執(zhí)行:
g++ -g -o test test.cpp –L/root/src/lib –ltest_so1
(此處,我們重點講解動態(tài)庫的動態(tài)調(diào)用的方法,關(guān)于靜態(tài)的通過g++編譯命令調(diào)用的方式不作詳細講解,具體相關(guān)內(nèi)容可上網(wǎng)查詢)
Linux下,提供專門的一組API用于完成打開動態(tài)庫,查找符號,處理出錯,關(guān)閉動態(tài)庫等功能。
下面對這些接口函數(shù)逐一介紹(調(diào)用這些接口時,需引用頭文件#include dlfcn.h):
1) dlopen
函數(shù)原型:void *dlopen(const char *libname,int flag);
功能描述:dlopen必須在dlerror,dlsym和dlclose之前調(diào)用,表示要將庫裝載到內(nèi)存,準備使用。如果要裝載的庫依賴于其它庫,必須首先裝載依賴庫。如果dlopen操作失敗,返回NULL值;如果庫已經(jīng)被裝載過,則dlopen會返回同樣的句柄。
參數(shù)中的libname一般是庫的全路徑,這樣dlopen會直接裝載該文件;如果只是指定了庫名稱,在dlopen會按照下面的機制去搜尋:
a.根據(jù)環(huán)境變量LD_LIBRARY_PATH查找
b.根據(jù)/etc/ld.so.cache查找
c.查找依次在/lib和/usr/lib目錄查找。
flag參數(shù)表示處理未定義函數(shù)的方式,可以使用RTLD_LAZY或RTLD_NOW。RTLD_LAZY表示暫時不去處理未定義函數(shù),先把庫裝載到內(nèi)存,等用到?jīng)]定義的函數(shù)再說;RTLD_NOW表示馬上檢查是否存在未定義的函數(shù),若存在,則dlopen以失敗告終。
2) dlerror
函數(shù)原型:char *dlerror(void);
功能描述:dlerror可以獲得最近一次dlopen,dlsym或dlclose操作的錯誤信息,返回NULL表示無錯誤。dlerror在返回錯誤信息的同時,也會清除錯誤信息。
3) dlsym
函數(shù)原型:void *dlsym(void *handle,const char *symbol);
功能描述:在dlopen之后,庫被裝載到內(nèi)存。dlsym可以獲得指定函數(shù)(symbol)在內(nèi)存中的位置(指針)。如果找不到指定函數(shù),則dlsym會返回NULL值。但判斷函數(shù)是否存在最好的方法是使用dlerror函數(shù),
4) dlclose
函數(shù)原型:int dlclose(void *);
功能描述:將已經(jīng)裝載的庫句柄減一,如果句柄減至零,則該庫會被卸載。如果存在析構(gòu)函數(shù),則在dlclose之后,析構(gòu)函數(shù)會被調(diào)用。
3、 普通函數(shù)的調(diào)用
此處以源碼實例說明。各源碼文件關(guān)系如下:
test_so1.h和test_so1.cpp生成test_so1.so動態(tài)庫。
test_so2.h和test_so2.cpp生成test_so2.so動態(tài)庫。
test_dl.cpp生成test_dl可執(zhí)行程序,test_dl通過dlopen系列等API函數(shù),并使用函數(shù)指針以到達動態(tài)調(diào)用不同so庫中test函數(shù)的目的。
-lxx
xx是你的.so文件名
其實使用方法和你使用數(shù)學(xué)庫函數(shù)是一樣的,源代碼中添加
#include math.h,編譯的時候,加上-lm參數(shù)。
注:linux下的.so文件為共享庫,相當(dāng)于windows下的dll文件。
擴展資料:?
linux下編寫調(diào)用so文件實例
.so是Linux(Unix)下的動態(tài)鏈接庫. 和.dll類似.
比如:
文件有: a.c, b.c, c.c
gcc -c a.c
gcc -c b.c
gcc -c c.c
gcc -shared libXXX.so a.o b.o c.o
要使用的話也很簡單. 比如編譯d.c, 使用到libXXX.so中的函數(shù), libXXX.so地址是MYPATH?
gcc d.c -o d -LMYPATH -lXXX
注意不是-llibXXX
test.c文件和一個test.h,這兩個文件要生成libsotest.so文件。然后我還有一個testso.c文件,在這個文件里面調(diào)用libsotest.so中的函數(shù)。
編寫的過程中,首先是編譯so文件,我沒有編寫makefile文件,而是參考的2里面說的直接寫的gcc命令。
因為so文件里面沒有main函數(shù),所以是不可執(zhí)行的,所以編譯的時候要加上-c,只生成目標文件。
本文名稱:linux命令加載.so linux加載文件命令
網(wǎng)址分享:http://sd-ha.com/article44/doocphe.html
成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供品牌網(wǎng)站建設(shè)、網(wǎng)站建設(shè)、企業(yè)網(wǎng)站制作、搜索引擎優(yōu)化、云服務(wù)器、自適應(yīng)網(wǎng)站
聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時需注明來源: 創(chuàng)新互聯(lián)