首先我們通過三個文件生成一個對應(yīng)的so庫毡代,然后演示如何調(diào)用對應(yīng)的庫劫拗。生成對應(yīng)的五個文件,test_a.c,test_b,c,test_c.c,so_test.h间校,test.c。
<code>
so_test.h
#include "stdio.h"
void test_a();
void test_b();
void test_c();
</code>
<code>
test_a.c
#include "so_test.h"
void test_a()
{
printf("this is in test_a..\n");
}
</code>
<code>
test_b.c
#include "so_test.h"
void test_b()
{
printf("this is in test_b..\n");
}
</code>
<code>
test_c.c
#include "so_test.h"
void test_c()
{
printf("this is in test_c..\n");
}
</code>
<code>
test.c
#include "so_test.h"
int main()
{
test_a();
test_b();
test_c();
return 0;
}
</code>
生成動態(tài)庫
編程三個文件生成so庫页慷,其中-shared指生成動態(tài)庫憔足,-fPIC指生成的庫地址無關(guān)。
<code>gcc test_a.c test_b.c test_c.c -fPIC -shared -o libtest.so</code>
使用動態(tài)庫
<code>gcc -o test test.c -L. -ltest</code>
但是此時如果使用ldd或者運(yùn)行test程序的話酒繁,就會發(fā)現(xiàn)程序還是運(yùn)行不了滓彰。
<code>crystal@crystal:~/workspace/sotest$ ldd test
linux-vdso.so.1 => (0x00007ffea3b8a000)
libtest.so => not found
libc.so.6 => /lib/x86_64-linux-gnu/libc.so.6 (0x00007f095aa14000)
/lib64/ld-linux-x86-64.so.2 (0x00005564bf151000)</code>
這個因為我們個人的so庫,并不能被系統(tǒng)直接識別欲逃,需要執(zhí)行LD_LIBRARY_PATH
或者在/etc/ld.so.conf.d目錄下面添加對應(yīng)的庫項找蜜。
執(zhí)行<code>export LD_LIBRARY_PATH=$(pwd)</code>然后重新運(yùn)行程序。但是每次都這樣會顯得很麻煩稳析,這是我們可以把我們放置so的目錄添加到/etc/ld.so.conf.d/目錄下面去,然后執(zhí)行ldconfig命令
<code>
crystal@crystal:~/workspace/sotest$ sudo vim /etc/ld.so.conf.d/test.conf
crystal@crystal:~/workspace/sotest$ cat /etc/ld.so.conf.d/test.conf
/home/crystal/workspace/sotest
crystal@crystal:~/workspace/sotest$ sudo ldconfig
crystal@crystal:~/workspace/sotest$ ./test
this is in test_a..
this is in test_b..
this is in test_c..</code>
生成靜態(tài)庫
生成對應(yīng)的三個.o文件
<code>gcc -c test_a.c test_b.c test_c.c </code>
生成libtest.a靜態(tài)庫
<code>ar rcs libtest.a test_a.o test_b.o test_c.o</code>
使用靜態(tài)庫
<code>gcc -o test test.c -static -L. -ltest</code>
然后運(yùn)行和查看程序
<code>crystal@crystal:~/workspace/sotest$ ./test
this is in test_a..
this is in test_b..
this is in test_c..
crystal@crystal:~/workspace/sotest$ ldd test
不是動態(tài)可執(zhí)行文件</code>
此時如果刪除libtest.a文件程序也是可以正常運(yùn)行的弓叛,并且test可執(zhí)行程序會比其他動態(tài)可執(zhí)行文件大很多彰居。
NOTE
****編譯參數(shù)解析
**
最主要的是GCC命令行的一個選項:-shared 該選項指定生成動態(tài)連接庫(讓連接器生成T類型的導(dǎo)出符號表,有時候也生成弱連接W類型的導(dǎo)出符號)撰筷,不用該標(biāo)志外部程序無法連接陈惰。相當(dāng)于一個可執(zhí)行文件
- -fPIC:表示編譯為位置獨立的代碼,不用此選項的話編譯后的代碼是位置相關(guān)的所以動態(tài)載入時是通過代碼拷貝的方式來滿足不同進(jìn)程的需要毕籽,而不能達(dá)到真正代碼段共享的目的抬闯。
- -L.:表示要連接的庫在當(dāng)前目錄中
- -ltest:編譯器查找動態(tài)連接庫時有隱含的命名規(guī)則,即在給出的名字前面加上lib关筒,后面加上.so來確定庫的名稱l LD_LIBRARY_PATH:這個環(huán)境變量指示動態(tài)連接器可以裝載動態(tài)庫的路徑溶握。
- 當(dāng)然如果有root權(quán)限的話,可以修改/etc/ld.so.conf文件蒸播,然后調(diào)用 /sbin/ldconfig來達(dá)到同樣的目的睡榆,不過如果沒有root權(quán)限,那么只能采用輸出LD_LIBRARY_PATH的方法了袍榆。
調(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)庫的目錄奴愉。通常這樣做就可以解決庫無法鏈接的問題了。
靜態(tài)庫鏈接時搜索路徑順序: - ld會去找GCC命令中的參數(shù)-L2. 再找gcc的環(huán)境變量LIBRARY_PATH3. 再找內(nèi)定目錄 /lib /usr/lib /usr/local/lib 這是當(dāng)初compile gcc時寫在程序內(nèi)的
動態(tài)鏈接時铁孵、執(zhí)行時搜索路徑順序: - 編譯目標(biāo)代碼時指定的動態(tài)庫搜索路徑锭硼;
- 環(huán)境變量LD_LIBRARY_PATH指定的動態(tài)庫搜索路徑;
- 配置文件/etc/ld.so.conf中指定的動態(tài)庫搜索路徑蜕劝;
- 默認(rèn)的動態(tài)庫搜索路徑/lib檀头;
- 默認(rèn)的動態(tài)庫搜索路徑/usr/lib。
有關(guān)環(huán)境變量:
LIBRARY_PATH環(huán)境變量:指定程序靜態(tài)鏈接庫文件搜索路徑LD_LIBRARY_PATH環(huán)境變量:指定程序動態(tài)鏈接庫文件搜索路徑