Linux下的靜態(tài)庫、動態(tài)庫和動態(tài)加載庫

作者 | P_Chou水冗

Linux庫類型

Linux下可以創(chuàng)建兩種類型的庫:

  • 靜態(tài)庫(.a): 在鏈接期間被應用程序直接鏈接進可執(zhí)行文件

  • 動態(tài)鏈接庫(.so): 動態(tài)庫還分為兩種用法: a) 應用程序運行期間鏈接動態(tài)庫萌朱,但是在編譯期間聲明動態(tài)庫的存在宴树,也就是說這種動態(tài)庫必須在編譯時對編譯器可見,但編譯器卻不將此種庫編譯進可執(zhí)行文件; b) 在運行期間晶疼,動態(tài)加載和卸載的庫酒贬,使用動態(tài)加載方法加載。這種庫的形式跟動態(tài)鏈接沒有本質(zhì)區(qū)別翠霍,區(qū)別是在調(diào)用時锭吨,是由用戶程序決定何時鏈接的,而不是由系統(tǒng)鏈接器自動鏈接

命名約定

庫需要以lib作為開頭寒匙,而在指定鏈接命令行參數(shù)時零如,卻無需包含開頭和擴展名,例如:

gcc src-file.c -lm -lpthread

這個例子中锄弱,鏈接了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

  • 構(gòu)建符號表。ranlib libctest.a用于為.a創(chuàng)建符號表葵姥。有些ar命令實際上已經(jīng)集成了ranlib的功能

  • a文件與windows下的.lib是相同的概念荷鼠。

動態(tài)庫(.so)

生成動態(tài)庫的方法如下:

  • 編譯object文件時使用-fPIC選項:
gcc -Wall -fPIC -c *.c

這個選項的目的是讓編譯器生成地址無關(guān)(position independent)的代碼,這是因為牌里,動態(tài)庫是在運行期間鏈接的颊咬,變量和函數(shù)的偏移量是事先不知道的,需要鏈接以后根據(jù)offset進行地址重定向牡辽。

  • 使用-shared鏈接
gcc -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)建兩個軟鏈:

mv libctest.so.1.0 /opt/lib
ln -sf /opt/lib/libctest.so.1.0 /opt/lib/libctest.so.1
ln -sf /opt/lib/libctest.so.1.0 /opt/lib/libctest.so

libctest.so用于在編譯期間使用-lctest讓編譯器找到動態(tài)庫,而libctest.so.1用于在運行期間鏈接

gcc -Wall -I/path/to/include-files -L/path/to/libraries prog.c -lctest -o prog

查看依賴

使用ldd命令來查看程序?qū)討B(tài)庫的依賴奏黑。例如:

ldd prog

libctest.so.1 => /opt/lib/libctest.so.1 (0x00002aaaaaaac000)
libc.so.6 => /lib64/tls/libc.so.6 (0x0000003aa4e00000)
/lib64/ld-linux-x86-64.so.2 (0x0000003aa4c00000)

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)加載和卸載的庫

需要應用程序希望設計成插件化的架構(gòu)计雌,這就需要可以動態(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編譯和鏈接會產(chǎn)生類似.la的文件,.la這種文件其實是個文本文件蛾洛,指向.a文件养铸,并聲明一些版本信息。

原文轉(zhuǎn)自:https://segmentfault.com/a/1190000005988462

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末轧膘,一起剝皮案震驚了整個濱河市钞螟,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌谎碍,老刑警劉巖鳞滨,帶你破解...
    沈念sama閱讀 217,509評論 6 504
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異蟆淀,居然都是意外死亡拯啦,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,806評論 3 394
  • 文/潘曉璐 我一進店門熔任,熙熙樓的掌柜王于貴愁眉苦臉地迎上來褒链,“玉大人,你說我怎么就攤上這事疑苔「ζィ” “怎么了?”我有些...
    開封第一講書人閱讀 163,875評論 0 354
  • 文/不壞的土叔 我叫張陵夯巷,是天一觀的道長赛惩。 經(jīng)常有香客問我,道長趁餐,這世上最難降的妖魔是什么喷兼? 我笑而不...
    開封第一講書人閱讀 58,441評論 1 293
  • 正文 為了忘掉前任,我火速辦了婚禮后雷,結(jié)果婚禮上季惯,老公的妹妹穿的比我還像新娘。我一直安慰自己臀突,他們只是感情好勉抓,可當我...
    茶點故事閱讀 67,488評論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著候学,像睡著了一般藕筋。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上梳码,一...
    開封第一講書人閱讀 51,365評論 1 302
  • 那天隐圾,我揣著相機與錄音伍掀,去河邊找鬼。 笑死暇藏,一個胖子當著我的面吹牛蜜笤,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播盐碱,決...
    沈念sama閱讀 40,190評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼把兔,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了瓮顽?” 一聲冷哼從身側(cè)響起县好,我...
    開封第一講書人閱讀 39,062評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎暖混,沒想到半個月后聘惦,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,500評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡儒恋,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,706評論 3 335
  • 正文 我和宋清朗相戀三年善绎,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片诫尽。...
    茶點故事閱讀 39,834評論 1 347
  • 序言:一個原本活蹦亂跳的男人離奇死亡禀酱,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出牧嫉,到底是詐尸還是另有隱情剂跟,我是刑警寧澤,帶...
    沈念sama閱讀 35,559評論 5 345
  • 正文 年R本政府宣布酣藻,位于F島的核電站曹洽,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏辽剧。R本人自食惡果不足惜送淆,卻給世界環(huán)境...
    茶點故事閱讀 41,167評論 3 328
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望怕轿。 院中可真熱鬧偷崩,春花似錦、人聲如沸撞羽。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,779評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽诀紊。三九已至谒出,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背笤喳。 一陣腳步聲響...
    開封第一講書人閱讀 32,912評論 1 269
  • 我被黑心中介騙來泰國打工考赛, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人莉测。 一個月前我還...
    沈念sama閱讀 47,958評論 2 370
  • 正文 我出身青樓,卻偏偏與公主長得像唧喉,于是被迫代替她去往敵國和親捣卤。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 44,779評論 2 354

推薦閱讀更多精彩內(nèi)容