新聞中心
在Linux下開發(fā)應(yīng)用程序時,我們經(jīng)常會使用動態(tài)鏈接庫(.so文件)來共享代碼,提高程序的執(zhí)行效率。SO接口函數(shù)是動態(tài)鏈接庫中公開的函數(shù),開發(fā)人員可以調(diào)用這些函數(shù)實(shí)現(xiàn)特定的功能。但是,有時候我們需要查看這些SO接口函數(shù)的定義,以便更好地理解和使用它們。本文將介紹Linux下查看SO接口函數(shù)的方法,相信讀完后你就可以輕松掌握這個技能了。

愛民網(wǎng)站制作公司哪家好,找創(chuàng)新互聯(lián)!從網(wǎng)頁設(shè)計、網(wǎng)站建設(shè)、微信開發(fā)、APP開發(fā)、響應(yīng)式網(wǎng)站開發(fā)等網(wǎng)站項(xiàng)目制作,到程序開發(fā),運(yùn)營維護(hù)。創(chuàng)新互聯(lián)于2013年成立到現(xiàn)在10年的時間,我們擁有了豐富的建站經(jīng)驗(yàn)和運(yùn)維經(jīng)驗(yàn),來保證我們的工作的順利進(jìn)行。專注于網(wǎng)站建設(shè)就選創(chuàng)新互聯(lián)。
一、使用nm命令查看SO庫函數(shù)
nm命令是Linux下的一個二進(jìn)制程序查看工具,用于查看可執(zhí)行文件和共享對象(即.so文件)中的符號表。nm命令有三個常用的選項(xiàng):-A、-a和-D,它們分別表示對所有符號、對所有符號和調(diào)試符號以及對所有動態(tài)符號表符號進(jìn)行列出。以下是使用nm命令查看SO庫函數(shù)的詳細(xì)步驟:
1. 打開終端,進(jìn)入SO庫所在的目錄
2. 輸入以下命令,列出這個SO庫中的所有符號及其地址:
nm -gC xxx.so
其中,-g表示顯示全局符號,-C表示顯示C++符號
3. 查找你要查看的函數(shù)名,如果該函數(shù)被定義在SO庫中,就會在列表中出現(xiàn),其地址就是函數(shù)名前面的十六進(jìn)制數(shù)值
4. 接下來可以查看該函數(shù)的定義,使用objdump命令,輸入以下命令:
objdump -D xxx.so | grep 函數(shù)名
其中,-D表示反匯編所有節(jié)(section)的內(nèi)容,然后使用grep命令過濾出包含該函數(shù)名的內(nèi)容。
二、使用readelf命令查看SO庫函數(shù)
readelf是一個Linux系統(tǒng)下的ELF格式可執(zhí)行文件的查看工具。ELF(Executable and Linkable Format)是Linux下二進(jìn)制文件格式的標(biāo)準(zhǔn),它包含了代碼、數(shù)據(jù)和符號表等多種信息。下面是使用readelf命令查看SO庫函數(shù)的方法:
1. 打開終端,進(jìn)入SO庫所在的目錄
2. 輸入以下命令,查看SO庫中的符號表:
readelf -s xxx.so
其中,-s表示顯示符號表
3. 查找你要查看的函數(shù)名,如果該函數(shù)被定義在SO庫中,它就會在列表中出現(xiàn),其地址就是第三列的十六進(jìn)制數(shù)值
4. 接下來可以查看該函數(shù)的定義,使用objdump命令,輸入以下命令:
objdump -D xxx.so | grep 函數(shù)名
其中,-D表示反匯編所有節(jié)(section)的內(nèi)容,然后使用grep命令過濾出包含該函數(shù)名的內(nèi)容。
三、使用nm、readelf和objdump聯(lián)合使用查看SO庫函數(shù)
以上兩種方法都可以查看SO庫中的符號表及函數(shù)地址,但是只能通過objdump命令查看函數(shù)的定義。如果想要更直接地查看函數(shù)的定義,那么可以將這三個命令聯(lián)合使用。以下是具體步驟:
1. 打開終端,進(jìn)入SO庫所在的目錄
2. 輸入以下命令,查看符號表:
nm -D xxx.so | grep 函數(shù)名
其中,-D表示顯示動態(tài)符號表,grep命令用于過濾出包含該函數(shù)名的條目。這個命令會顯示出函數(shù)的地址和符號名。
3. 接下來使用以下命令,查看函數(shù)的定義:
objdump -D xxx.so | grep -A20 -B20 函數(shù)名
其中,-A20表示顯示函數(shù)名后面20行對象的內(nèi)容,-B20表示顯示函數(shù)名前面20行的內(nèi)容。這樣就可以直接查看函數(shù)的定義了。
以上就是在Linux下查看SO接口函數(shù)的方法,希望對你有所幫助。不過需要注意的是,這種查看方法只適用于開發(fā)人員,在日常使用中更好不要動SO文件、靜態(tài)庫文件、可執(zhí)行文件等,否則可能會導(dǎo)致系統(tǒng)崩潰。
成都網(wǎng)站建設(shè)公司-創(chuàng)新互聯(lián)為您提供網(wǎng)站建設(shè)、網(wǎng)站制作、網(wǎng)頁設(shè)計及定制高端網(wǎng)站建設(shè)服務(wù)!
請教關(guān)于android linux動態(tài)庫.so的加載調(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
(此處,我們重點(diǎn)講解動態(tài)庫的動態(tài)調(diào)用的方法,關(guān)于靜態(tài)的通過g++編譯命令調(diào)用的方式不作詳細(xì)講解,具體相關(guān)內(nèi)容可上網(wǎng)查詢)
Linux下,提供專門的一組API用于完成打開動態(tài)庫慧蘆,查找符號,處理出錯,關(guān)閉動態(tài)庫等功能。
下面對這些接口函數(shù)逐一介紹(調(diào)用這些接口時,需引用頭文件#include ):
1)dlopen
函數(shù)原型:void *dlopen(const char *libname,int flag);
功能描述:dlopen必須在dlerror,dlsym和dlclose之前調(diào)用,表示要將庫裝載到內(nèi)存,準(zhǔn)備使用。如果要裝載的庫依賴于其它庫,必須首先裝載依賴庫。如果dlopen操作失敗,返回NULL值;如果庫已經(jīng)被裝載過,則dlopen會返回同樣的句柄。
參數(shù)中的libname一般是庫的全路徑,這樣dlopen會直接裝載該文件;如果只是指定了庫名稱,在dlopen會按照下面的機(jī)制去搜尋:
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)用
此處以源碼實(shí)例說明。各源碼文件關(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ù)指針以到達(dá)動態(tài)調(diào)用不同so庫中test函數(shù)的目的。
請問我有一個.so文件,如何在Linux下編程使用呢?
Linux下的.so是基于linux下的動態(tài)鏈接,其功能和作用類似與windows下.dll文件。
下面是關(guān)于.so的介紹:
一、引言
通常情況下,對函數(shù)庫的鏈接是放在編譯時期(compile time)完成的。所有相關(guān)的對象文件(object file)與牽涉到的函數(shù)庫(library)被鏈接合成一個可執(zhí)行文件(executable file)。程序在運(yùn)行時,與函數(shù)庫再無瓜葛,因?yàn)樗行枰暮瘮?shù)已拷貝到自己門下。所以這些函數(shù)庫被成為靜態(tài)庫(static libaray),通常文件名為“l(fā)ibxxx.a”的形式。
其實(shí),我們也可以把對一些庫函數(shù)的鏈接載入推遲到程序運(yùn)行的時期(runtime)。這就是如雷貫耳的動態(tài)鏈接庫(dynamic link library)技術(shù)。
二、動態(tài)鏈接庫的特點(diǎn)與優(yōu)勢
首先讓我們來看一下,把庫函數(shù)推遲到程序運(yùn)行時期載入的好處:
1. 可以實(shí)現(xiàn)進(jìn)程之間的資源共享。
什么概念呢?就是說,某個程序的在運(yùn)行中要調(diào)用某個動態(tài)鏈接庫函數(shù)的時候,操作系統(tǒng)首先會查看所有正在運(yùn)行的程序,看在內(nèi)存里是否已有此庫函數(shù)的拷貝了。如果有,則讓其共享那一個拷貝;只有沒有才鏈接載入。這樣的模式雖然會帶來一些“動態(tài)鏈接”額外的開銷,卻大大的節(jié)省了系統(tǒng)的內(nèi)存資源。C的標(biāo)準(zhǔn)庫就是動態(tài)鏈接庫,也就是說系統(tǒng)中所有運(yùn)行的程序共享著同一個C標(biāo)準(zhǔn)庫的代碼段。
2. 將一些程序升級變得簡單。用戶只需要升級動態(tài)鏈接庫,而無需重新編譯鏈接其他原有的代碼就可以完成整個程序的升級。Windows 就是一個很好的例子。
3. 甚至可以真正坐到鏈接載入完全由程序員在程序代碼中控制。
程序員在編寫程序的時候,可以明確的指明什么時候或者什么情況下,鏈接載入哪個動態(tài)鏈接庫函數(shù)。你可以有一個相當(dāng)大的軟件,但每次運(yùn)行的時候,由于不同的操作需求,只有一小部分程序被載入內(nèi)存。所有的函數(shù)本著“有需求才調(diào)入”的原則,于是大大節(jié)省了系統(tǒng)資源。比如現(xiàn)在的軟件通常都能打開若干種不同類型的文件,這些讀寫操作通常都用動態(tài)鏈接庫來實(shí)現(xiàn)。在一次運(yùn)行當(dāng)中,一般只有一種類型的文件將會被打開。所以直到程序知道文件的類型以后再載入相應(yīng)的讀寫函數(shù),而不是一開始就將所有的讀寫函數(shù)都載入,然后才發(fā)覺在整個程序中根本沒有用到它們。
三、動態(tài)鏈接庫的創(chuàng)建
由于動態(tài)鏈接庫函數(shù)的共享特性,它們不會被拷貝到可執(zhí)行文件中。在編譯的時候,編譯器只會做一些函數(shù)名之類的檢查。在程序運(yùn)行的時候,被調(diào)用的動態(tài)鏈接庫函數(shù)被安置在內(nèi)存的某個地方,所有調(diào)用它的程序?qū)⒅赶蜻@個代碼段。因此,這些代碼必須實(shí)用相對地址,而不是絕對地址。在編譯的時候,我們需要告訴編譯器,這些對象文件是用來做動態(tài)鏈接庫的,所以要用地址不無關(guān)代碼(Position Independent Code (PIC))。
對gcc編譯器,只需添加上 -fPIC 標(biāo)簽,如:
gcc -fPIC -c file1.c
gcc -fPIC -c file2.c
gcc -shared libxxx.so file1.o file2.o
注意到最后一行,-shared 標(biāo)簽告訴編譯器這是要建立動態(tài)鏈接庫。這與靜態(tài)鏈接庫的建立很不一樣,后者用的是 ar 命令。也注意到,動態(tài)鏈接庫的名字形式為 “l(fā)ibxxx.so” 后綴名為 “.so”
四、動態(tài)鏈接庫的使用
使用動態(tài)鏈接庫,首先需要在編譯期間讓編譯器檢查一些語法與定義。
這與靜態(tài)庫的實(shí)用基本一樣,用的是 -Lpath 和 -lxxx 標(biāo)簽。如:
gcc file1.o file2.o -Lpath -lxxx -o program.exe
編譯器會先在path文件夾下搜索libxxx.so文件,如果沒有找到,繼續(xù)搜索libxxx.a(靜態(tài)庫)。
在程序運(yùn)行期間,也需要告訴系統(tǒng)去哪里找你的動態(tài)鏈接庫文件。在UNIX下是通過定義名為 LD_LIBRARY_PATH 的環(huán)境變量來實(shí)現(xiàn)的。只需將path賦值給此變量即可。csh 命令為:
setenv LD_LIBRARY_PATH your/full/path/to/dll
一切安排妥當(dāng)后,你可以用 ldd 命令檢查是否連接正常。
ldd program.exe
動態(tài)鏈接庫*.so的編譯與使用- –
動態(tài)庫*.so在linux下用c和c++編程時經(jīng)常會碰到,最近在網(wǎng)站找了幾篇文章介紹動態(tài)庫的編譯和鏈接,總算搞懂了這個之前一直不太了解得東東,這里做個筆記,也為其它正為動態(tài)庫鏈接庫而苦惱的兄弟們提供一點(diǎn)幫助。
1、動態(tài)庫的編譯
下面通過一個例子來介紹如何生成一個動態(tài)庫。這里有一個頭文件:so_test.h,三個.c文件:test_a.c、test_b.c、test_c.c,我們將這幾個文件編譯成一個動態(tài)庫:libtest.so。
so_test.h:
#include “stdio.h”
void test_a();
void test_b();
void test_c();
test_a.c:
#include “so_test.h”
void test_a()
{
printf(“this is in test_a…/n”);
}
test_b.c:
#include “so_test.h”
void test_b()
{
printf(“this is in test_b…/n”);
}
test_a.c:
#include “so_test.h”
void test_c()
{
printf(“this is in test_c…/n”);
}
將這幾個文件編譯成一個動態(tài)庫:libtest.so
$ gcc test_a.c test_b.c test_c.c -fPIC -shared -o libtest.so
2、動態(tài)庫的鏈接
在1、中,我們已經(jīng)成功生成了一個自己的動態(tài)鏈接庫libtest.so,下面我們通過一個程序來調(diào)用這個庫里的函數(shù)。程序的源文件為:test.c。
test.c:
#include “so_test.h”
int main()
{
test_a();
test_b();
test_c();
return 0;
}
l 將test.c與動態(tài)庫libtest.so鏈接生成執(zhí)行文件test:
$ gcc test.c -L. -ltest -o test
l 測試是否動態(tài)連接,如果列出libtest.so,那么應(yīng)該是連接正常了
$ ldd test
l 執(zhí)行test,可以看到它是如何調(diào)用動態(tài)庫中的函數(shù)的。
3、編譯參數(shù)解析
最主要的是GCC命令行的一個選項(xiàng):
-shared 該選項(xiàng)指定生成動態(tài)連接庫(讓連接器生成T類型的導(dǎo)出符號表,有時候也生成弱連接W類型的導(dǎo)出符號),不用該標(biāo)志外部程序無法連接。相當(dāng)于一個可執(zhí)行文件
l -fPIC:表示編譯為位置獨(dú)立的代碼,不用此選項(xiàng)的話編譯后的代碼是位置相關(guān)的所以動態(tài)載入時是通過代碼拷貝的方式來滿足不同進(jìn)程的需要,而不能達(dá)到真正代碼段共享的目的。
l -L.:表示要連接的庫在當(dāng)前目錄中
l -ltest:編譯器查找動態(tài)連接庫時有隱含的命名規(guī)則,即在給出的名字前面加上lib,后面加上.so來確定庫的名稱
l LD_LIBRARY_PATH:這個環(huán)境變量指示動態(tài)連接器可以裝載動態(tài)庫的路徑。
l 當(dāng)然如果有root權(quán)限的話,可以修改/etc/ld.so.conf文件,然后調(diào)用 /in/ldconfig來達(dá)到同樣的目的,不過如果沒有root權(quán)限,那么只能采用輸出LD_LIBRARY_PATH的方法了。
4、注意
調(diào)用動態(tài)庫的時候有幾個問題會經(jīng)常碰到,有時,明明已經(jīng)將庫的頭文件所在目錄 通過 “-I” include進(jìn)來了,庫所在文件通過 “-L”參數(shù)引導(dǎo),并指定了“-l”的庫名,但通過ldd命令察看時,就是死活找不到你指定鏈接的so文件,這時你要作的就是通過修改 LD_LIBRARY_PATH或者/etc/ld.so.conf文件來指定動態(tài)庫的目錄。通常這樣做就可以解決庫無法鏈接的問題了。
-lxx
xx是你的.so文件名
其實(shí)使用方法和你使用數(shù)學(xué)庫函數(shù)是一樣的,
源代碼
中添加
#include ,編譯的時候,加上-lm參數(shù)。
注:linux下的.so文件為共享庫,相當(dāng)于windows下的dll文件。
擴(kuò)展資料:
linux下編寫調(diào)用so文件實(shí)例
.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命令。
因?yàn)閟o文件里面沒有
main函數(shù)
,所以是不可執(zhí)行的,所以編譯的時候要加上-c,只生成目標(biāo)文件。
linux下的.so文件為共享庫,相當(dāng)于windows下的dll文件,使用方法如下:
在你的工程源代碼里包含.h頭文件,然后可以調(diào)用動態(tài)庫里的函數(shù),在鏈接的時候加上如下編譯器參數(shù):
-l xx.so
如果你的so文件是以lib開頭的,還可以直接這樣使用:
-lxx
xx是你的.so文件名
其實(shí)使用方法和你使用數(shù)學(xué)庫函數(shù)是一樣的,源代碼中添加
#include ,編譯的時候,加上-lm參數(shù)。
動態(tài)鏈接庫,調(diào)用,寫c語言時用的。放在編譯的文件夾里面。包含進(jìn)去.h就行了、、、、、、、、
安裝個開發(fā)工具 然后編譯就行了 redhat在安裝的時候選擇自定義就可以安裝開發(fā)工具 make 然后make install
關(guān)于linux查看so接口函數(shù)的介紹到此就結(jié)束了,不知道你從中找到你需要的信息了嗎 ?如果你還想了解更多這方面的信息,記得收藏關(guān)注本站。
香港云服務(wù)器機(jī)房,創(chuàng)新互聯(lián)(www.cdcxhl.com)專業(yè)云服務(wù)器廠商,回大陸優(yōu)化帶寬,安全/穩(wěn)定/低延遲.創(chuàng)新互聯(lián)助力企業(yè)出海業(yè)務(wù),提供一站式解決方案。香港服務(wù)器-免備案低延遲-雙向CN2+BGP極速互訪!
網(wǎng)頁題目:輕松學(xué)會Linux下查看SO接口函數(shù)方法(linux查看so接口函數(shù))
當(dāng)前地址:http://m.fisionsoft.com.cn/article/djdgjgp.html


咨詢
建站咨詢
