庫的存在極大的提高了C/C++程序的復用性,但是庫對于初學者來說有些難以駕馭,本文從Linux的角度淺談Linux下的靜態(tài)庫、動態(tài)庫和動態(tài)加載庫。
Linux庫類型
Linux下可以創(chuàng)建兩種類型的庫:
命名約定
庫需要以lib作為開頭,而在指定鏈接命令行參數(shù)時,卻無需包含開頭和擴展名,例如:
這個例子中,鏈接了libmath.a和libpthread.a
靜態(tài)庫(.a)
生成靜態(tài)庫的方法如下:
- 編譯object文件。例如:cc -Wall -c ctest1.c ctest2.c,該命令會生成ctest1.o和ctest2.o(其中-Wall表示編譯時輸出警告)。
- 創(chuàng)建庫文件。例如:ar -cvq libctest.a ctest1.o ctest2.o。該命令會得到一個libctest.a文件
- 可以通過ar -t查看.a文件中包含哪些.o。所以,實際上ar就是一個打包命令,類似tar
- 構建符號表。ranlib libctest.a用于為.a創(chuàng)建符號表。有些ar命令實際上已經集成了ranlib的功能
.a文件與windows下的.lib是相同的概念。
動態(tài)庫(.so)
生成動態(tài)庫的方法如下:
- 編譯object文件時使用-fPIC選項:?????Shell??gcc -Wall -fPIC -c *.c1gcc -Wall -fPIC -c *.c
這個選項的目的是讓編譯器生成地址無關(position independent)的代碼,這是因為,動態(tài)庫是在運行期間鏈接的,變量和函數(shù)的偏移量是事先不知道的,需要鏈接以后根據(jù)offset進行地址重定向。
- 使用-shared鏈接?????Shell??gcc -shared -Wl,-soname,libctest.so.1 -o libctest.so.1.0 *.o1gcc -shared -Wl,-soname,libctest.so.1 -o libctest.so.1.0 *.o
-shared選項是讓動態(tài)庫得以在運行期間被動態(tài)鏈接;-Wl,options是設置傳遞給ld(鏈接器)的參數(shù),在上面的例子中,當鏈接器在鏈接.o時會執(zhí)行l(wèi)d -soname ibctest.so.1
- 創(chuàng)建軟鏈
上面的命令將最終輸出一個動態(tài)庫libctest.so.1.0,而出于習慣,會創(chuàng)建兩個軟鏈:
libctest.so用于在編譯期間使用-lctest讓編譯器找到動態(tài)庫,而libctest.so.1用于在運行期間鏈接
?
查看依賴
使用ldd命令來查看程序對動態(tài)庫的依賴。例如:
?
obj文件
obj文件的格式和組成可能是系統(tǒng)差異性的一大體現(xiàn),比如windows下的PE、linux和一些unix下的elf、macos的mach-o、aix下的xcoff。
查看obj文件的符號表信息,可以通過nm objdump readelf等方法。
運行期間查找動態(tài)庫
運行期間,系統(tǒng)需要知道到哪里去查找動態(tài)庫,這是通過/etc/ld.so.conf配置的。ldconfig用于配置運行時動態(tài)庫查找路徑,實際是更新/etc/ld.so.cache。另外一些環(huán)境變量也可以影響查找:(Linux/Solaris: LD_LIBRARY_PATH, SGI: LD_LIBRARYN32_PATH, AIX: LIBPATH, Mac OS X: DYLD_LIBRARY_PATH, HP-UX: SHLIB_PATH)
動態(tài)加載和卸載的庫
需要應用程序希望設計成插件化的架構,這就需要可以動態(tài)加載和卸載庫的機制。與動態(tài)鏈接不同的是,動態(tài)加載的意思是,編譯期間可以對動態(tài)庫的存在一無所知,而是在運行期間通過用戶程序嘗試加載進來的。
通過dlfcn.h中的dlopen、dlsym和dlclose等函數(shù)實現(xiàn)此種功能。
另外,使用到dlfcn機制的可執(zhí)行文件需要使用-rdynamic選項,它將指示連接器把所有符號(而不僅僅只是程序已使用到的外部符號,但不包括靜態(tài)符號,比如被static修飾的函數(shù))都添加到動態(tài)符號表(即.dynsym表)里。
GNU Libtool
如今許多軟件的編譯都采用libtool工具,libtool是一個編譯鏈接包裝工具,實際只是一個腳本,用libtool編譯和鏈接會產生類似.la的文件,.la這種文件其實是個文本文件,指向.a文件,并聲明一些版本信息。
評論
查看更多