閱讀60 返回首頁    go 阿裏雲 go 技術社區[雲棲]


Linux靜態庫和動態庫的編寫和使用

Linux靜態庫和動態庫的編寫和使用
庫從本質上來說是一種可執行代碼的二進製格式,可以被載入內存中執行。庫分靜態庫和動態庫兩種。  
1  靜態庫和動態庫的區別
1.1. 靜態函數庫
   (1)靜態函數庫的名字一般是lib[name].a
(2)利用靜態函數庫編譯成的文件比較大,因為整個函數庫的所有數據都會被整合進目標代碼中,它的優點是編譯後的執行程序不需要外部的函數庫支持,因為所有使用的函數都已經被編譯進去了。這給它帶來的缺點為如果靜態函數庫改變了,那麼你的程序必須重新編譯。
1.2. 動態函數庫
   (1)動態函數庫的名字一般是lib[name].so
(2)相對於靜態函數庫,動態函數庫所產生的可執行文件比較小,動態函數庫在編譯的時候 並沒有被編譯進目標代碼中,程序執行到相關函數時才調用該函數庫裏的相應函數,因此動態函數庫所產生的可執行文件比較小。由於函數庫沒有被整合進你的 程序,而是程序運行時動態的申請並調用,所以程序的運行環境中必須提供相應的庫。動態函數庫的改變並不影響你的程序,所以動態函數庫的升級比較方便。
   (3)linux係統有幾個重要的目錄存放相應的函數庫,如/lib /usr/lib。
2   編寫及使用靜態庫  
靜態庫的製作所要用到的命令:gcc和ar 命令。  
2.1 編寫靜態庫的步驟為:
   2.1.1  創建編輯源代碼文件: pr1.c 和 pr2.c  
pr1.c:  
void print1()  
{  
                printf("This is the first lib src!\n");  
}  
pr2.c  
void print2()  
{  
                printf("This is the second src lib!\n");  
}  
      2.1.2  編譯.c 文件  
gcc  -O -c   pr1.c pr2.c  
2.1.3 鏈接靜態庫
在編譯程序中正確找到庫文件,靜態庫必須按照 lib[name].a 的規則命名,如下例中[name]=pr.  
(1)ar -rsv libpr.a pr1.o pr2.o  
a - pr1.o  
a - pr2.o  
(2) ar  t  ibpr.a  
pr1.o  
pr2.o  
2.1.4 調用庫函數代碼 main.c  
main.c  
int main()  
{  
                print1();  
                print2();  
                return 0;  
}  
2.1.5   編譯鏈接選項  
    -L 及-l 參數放在後麵.其中-L 加載庫文件路徑,-l 指明庫文件名字.  
gcc -o main main.c -L./ -lpr  
 2.1.6 執行目標程序  
3   編寫動態庫
3.1 設計庫代碼  
pr1.c  :
int p = 2;  
void print()
{  
                printf("This is the first dll src!\n");  
}      
3. 2 生成動態庫  
gcc -O -fpic -shared -o dl.so pr1.c  
4  動態庫的使用
4.1 動態庫的隱式調用  
在編譯調用庫函數代碼時指明動態庫的位置及名字
main.c  :
int main()  
{  
                print();  
                return 0;  
}  
gcc -o tdl main.c ./dl.so  
當動態庫的位置或者名字發生改變時,  程序將無法正常運行;  而動態庫取代靜態庫的好處之一則是通過更新動態庫而隨時升級庫的內容.  
4.2  動態庫的顯式調用  
     顯式調用動態庫需要四個函數的支持,  
(1)函數 dlopen 打開動態庫,  
(2)函數 dlsym 獲取動態庫中對象基址,  
(3)函數 dlerror 獲取顯式動態庫操作中的錯誤信息,  
(4)函數 doclose 關閉動態庫.
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;  
}  

  gcc -o tds main.c –ld1 –L.
此 時還不能立即./tds,因為在動態函數庫使用時,會查找/usr/lib、/lib目錄下的動態函數庫,而此時我們生成的庫不在裏邊。
解決方法有:
(1)最直接最簡單的方法就是把生成的動態鏈接庫放到/usr/lib或/lib中去。
(2) export LD_LIBRARY_PATH=$(pwd)
(3)另外還可以在/etc/ld.so.conf文件裏加入我們生成的庫的目錄,然後/sbin/ldconfig。 /etc/ld.so.conf是非常重要的一個目錄,裏麵存放的是鏈接器和加載器搜索共享庫時要檢查的目錄,默認是從/usr/lib /lib中讀取的,所以想要順利運行,我們也可以把我們庫的目錄加入到這個文件中並執行/sbin/ldconfig 。另外還有個文件需要了解/etc/ld.so.cache,裏麵保存了常用的動態函數庫,且會先把他們加載到內存中,因為內存的訪問速度遠遠大於硬盤的 訪問速度,這樣可以提高軟件加載動態函數庫的速度了。
5  庫依賴的查看
使用ldd命令來查看執行文件依賴於哪些庫。
該命令用於判斷某個可執行的 binary 檔案含有什麼動態函式庫。
 ldd [-vdr] [filename]
參數說明:
--version  打印ldd的版本號
-v --verbose  打印所有信息,例如包括符號的版本信息
-d --data-relocs  執行符號重部署,並報告缺少的目標對象(隻對ELF格式適用)
-r --function-relocs  對目標對象和函數執行重新部署,並報告缺少的目標對象和函數(隻對ELF格式適用)
--help 用法信息。
如果命令行中給定的庫名字包含'/',這個程序的libc5版本將使用它作為庫名字;否則它將在標準位置搜索庫。運行一個當前目錄下的共享庫,加前綴"./"。



最後更新:2017-04-02 15:28:25

  上一篇:go Linux下多進程編程(C語言)
  下一篇:go ListView中getView中放置多個item和getItemViewType的用法