10.03 讓我們來搞懂Linux靜態庫和動態庫

庫從本質上來說是一種可執行代碼的二進制格式,可以被載入內存中執行。庫分靜態庫和動態庫兩種。

靜態庫和動態庫的區別

1. 靜態函數庫

這類庫的名字一般是libxxx.a;利用靜態函數庫編譯成的文件比較大,因為整個 函數庫的所有數據都會被整合進目標代碼中,他的優點就顯而易見了,即編譯後的執行程序不需要外部的函數庫支持,因為所有使用的函數都已經被編譯進去了。當然這也會成為他的缺點,因為如果靜態函數庫改變了,那麼你的程序必須重新編譯。

2. 動態函數庫

這類庫的名字一般是libxxx.so;相對於靜態函數庫,動態函數庫在編譯的時候 並沒有被編譯進目標代碼中,你的程序執行到相關函數時才調用該函數庫裡的相應函數,因此動態函數庫所產生的可執行文件比較小。由於函數庫沒有被整合進你的程序,而是程序運行時動態的申請並調用,所以程序的運行環境中必須提供相應的庫。動態函數庫的改變並不影響你的程序,所以動態函數庫的升級比較方便。

linux系統有幾個重要的目錄存放相應的函數庫,如/lib /usr/lib。

靜態庫的使用

靜態庫的操作工具:gcc和ar 命令。

編寫及使用靜態庫

(1)設計庫源碼 pr1.c 和 pr2.c

[root@billstone make_lib]# cat pr1.c 
void print1()
{
printf("This is the first lib src!\\n");
}
[root@billstone make_lib]# cat pr2.c
void print2()
{
printf("This is the second src lib!\\n");
}

(2) 編譯.c 文件

[bill@billstone make_lib]$ cc -O -c pr1.c pr2.c 
[bill@billstone make_lib]$ ls -l pr*.o
-rw-rw-r-- 1 bill bill 804 4 月 15 11:11 pr1.o
-rw-rw-r-- 1 bill bill 804 4 月 15 11:11 pr2.o

(3) 鏈接靜態庫

 為了在編譯程序中正確找到庫文件,靜態庫必須按照 lib[name].a 的規則命名,如下例中[name]=pr. 
[bill@billstone make_lib]$ ar -rsv libpr.a pr1.o pr2.o
a - pr1.o

a - pr2.o
[bill@billstone make_lib]$ ls -l *.a
-rw-rw-r-- 1 bill bill 1822 4 月 15 11:12 libpr.a
[bill@billstone make_lib]$ ar -t libpr.a
pr1.o
pr2.o

(4) 調用庫函數代碼 main.c

[bill@billstone make_lib]$ cat main.c 
int main()
{
print1();
print2();
return 0;
}

(5) 編譯鏈接選項

-L 及-l 參數放在後面.其中,-L 加載庫文件路徑,-l 指明庫文件名字.

[bill@billstone make_lib]$ gcc -o main main.c -L./ -lpr 
[bill@billstone make_lib]$ ls -l main*
-rwxrwxr-x 1 bill bill 11805 4 月 15 11:17 main
-rw-rw-r-- 1 bill bill 50 4 月 15 11:15 main.c

(6)執行目標程序

[bill@billstone make_lib]$ ./main 
This is the first lib src!
This is the second src lib!
[bill@billstone make_lib]$

動態庫的使用

編寫動態庫

(1)設計庫代碼

[bill@billstone make_lib]$ cat pr1.c 
int p = 2;
void print(){
printf("This is the first dll src!\\n");
}
[bill@billstone make_lib]$

(2)生成動態庫

[bill@billstone make_lib]$ gcc -O -fpic -shared -o dl.so pr1.c 
[bill@billstone make_lib]$ ls -l *.so
-rwxrwxr-x 1 bill bill 6592 4 月 15 15:19 dl.so
[bill@billstone make_lib]$

動態庫的隱式調用

在編譯調用庫函數代碼時指明動態庫的位置及名字, 看下面實例

[bill@billstone make_lib]$ cat main.c 
int main()
{
print();
return 0;
}
[bill@billstone make_lib]$ gcc -o tdl main.c ./dl.so
[bill@billstone make_lib]$ ./tdl
This is the first dll src!
[bill@billstone make_lib]$

當動態庫的位置活名字發生改變時, 程序將無法正常運行; 而動態庫取代靜態庫的好處之一則是通過更新動態庫而隨時升級庫的內容.

動態庫的顯式調用

顯式調用動態庫需要四個函數的支持, 函數 dlopen 打開動態庫, 函數 dlsym 獲取動態庫中對象基址, 函數 dlerror 獲取顯式動態庫操作中的錯誤信息, 函數 doclose 關閉動態庫.

[bill@billstone make_lib]$ cat main.c 
#include <dlfcn.h>
int main()
{
void *pHandle;
void (*pFunc)(); // 指向函數的指針
int *p;
pHandle = dlopen("./d1.so", RTLD_NOW); // 打開動態庫
if(!pHandle){
printf("Can't find d1.so \\n");
exit(1);
}
pFunc = (void (*)())dlsym(pHandle, "print"); // 獲取庫函數 print 的地址
if(pFunc)
pFunc();
else
printf("Can't find function print\\n");
p = (int *)dlsym(pHandle, "p"); // 獲取庫變量 p 的地址
if(p)
printf("p = %d\\n", *p);
else
printf("Can't find int p\\n");
dlclose(pHandle); // 關閉動態庫
return 0;
}
[bill@billstone make_lib]$ gcc -o tds main.c –ld1 –L.
/<dlfcn.h>

此時還不能立即./tds,因為在動態函數庫使用時,會查找/usr/lib、/lib目錄下的動態函數庫,而此時我們生成的庫不在裡邊。 這個時候有好幾種方法可以讓他成功運行: 最直接最簡單的方法就是把libstr_out.so拉到/usr/lib或/lib中去。 還有一種方法 export LD_LIBRARY_PATH=$(pwd) 另外還可以在/etc/ld.so.conf文件里加入我們生成的庫的目錄,然後/sbin/ldconfig。 /etc/ld.so.conf是非常重要的一個目錄,裡面存放的是鏈接器和加載器搜索共享庫時要檢查的目錄,默認是從/usr/lib /lib中讀取的,所以想要順利運行,我們也可以把我們庫的目錄加入到這個文件中並執行/sbin/ldconfig 。另外還有個文件需要了解/etc/ld.so.cache,裡面保存了常用的動態函數庫,且會先把他們加載到內存中,因為內存的訪問速度遠遠大於硬盤的訪問速度,這樣可以提高軟件加載動態函數庫的速度了。

庫依賴的查看

使用ldd命令來查看執行文件依賴於哪些庫。

該命令用於判斷某個可執行的 binary 檔案含有什麼動態函式庫。

[root@test root]# ldd [-vdr] [filename]

參數說明:

--version 打印ldd的版本號

-v --verbose 打印所有信息,例如包括符號的版本信息

-d --data-relocs 執行符號重部署,並報告缺少的目標對象(只對ELF格式適用)

-r --function-relocs 對目標對象和函數執行重新部署,並報告缺少的目標對象和函數(只對ELF格式適用)

--help 用法信息。

如果命令行中給定的庫名字包含'/',這個程序的libc5版本將使用它作為庫名字;否則它將在標準位置搜索庫。運行一個當前目錄下的共享庫,加前綴"./"。


分享到:


相關文章: