Linux中有兩類函數庫,分別是靜態庫和動態庫。
靜態函數庫:
這類庫的名字一般是libxxx.a;利用靜態函數庫編譯成的文件比較大,因為整個函數庫的所有數據都會被整合進目標代碼中,他的優點就顯而易見了,即編譯後的執行程序不需要外部的函數庫支持,因為所有使用的函數都已經被編譯進去了。當然這也會成為他的缺點,因為如果靜態函數庫改變了,那麼你的程序必須重新編譯。
動態函數庫:
這類庫的名字一般是libxxx.so;相對於靜態函數庫,動態函數庫在編譯的時候並沒有被編譯進目標代碼中,你的程序執行到相關函數時才調用該函數庫裡的相應函數,因此動態函數庫所產生的可執行文件比較小。由於函數庫沒有被整合進你的程序,而是程序運行時動態的申請並調用,所以程序的運行環境中必須提供相應的庫。動態函數庫的改變並不影響你的程序,所以動態函數庫的升級比較方便。
linux系統有幾個重要的目錄存放相應的函數庫,如/lib /usr/lib.
下面來介紹linux靜態函數庫的創建和使用:
例程str_out.h str_out.c main.c:
str_out.h
#ifndef STR_OUT_H
#define STR_OUT_H
void str_out(const char* str);
#endif
str_out.c
#include
#include "str_out.h"
void str_out(const char* str){
printf("%s\n",str);
}
main.c
int main()
{
str_out("hello world");
return 0;
}
不管是靜態函數庫還是動態函數庫,都是由*.o目標文件生成。
所以先gcc -c str_out.c
靜態函數庫由ar命令創建
本例:ar -cr libstr_out.a str_out.o
-c create的意思
-r replace的意思,表示當插入的模塊名已經在庫中存在,則替換同名的模塊。如果若干模塊中有一個模塊在庫中不存在,ar顯示一個錯誤消息,並不替換其他同名模塊。默認的情況下,新的成員增加在庫的結尾處,可以使用其他任選項來改變增加的位置。
到此靜態函數庫創建完畢。
使用方法:通過gcc -o out main.c -L. -lstr_out編譯main.c就會把靜態函數庫整合進out.
其中
-L指定靜態函數庫的位置供查找,注意L後面還有'.',表示靜態函數庫在本目錄下查找。
-l則指定了靜態函數庫名,由於靜態函數庫的命名方式是lib***.a,其中的lib和。a忽略。
根據靜態函數庫的特性,此處刪除libstr_out.a後out依然可以運行,因為靜態庫的內容已經整合進去了。
動態函數庫的創建和使用
gcc -shared -fPCI -o out main.c -L. -lstr_out
用該命令生成libstr_out.so 動態函數庫。
gcc -o out main.c
此時還不能立即。/out,因為在動態函數庫使用時,會查找/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,裡面保存了常用的動態函數庫,且會先把他們加載到內存中,因為內存的訪問速度遠遠大於硬盤的訪問速度,這樣可以提高軟件加載動態函數庫的速度了