最近在Linux下使用第三方庫(kù)Protobuf時(shí),遇到一個(gè)問(wèn)題:可執(zhí)行程序在運(yùn)行時(shí)報(bào)錯(cuò):“error while loading shared libraries: libprotobuf.so.7: cannot open shared object file: No such file or directory”。于是花時(shí)間弄清楚原因,找到解決方案,跟大家共享一下。
1. 什么是庫(kù)
在windows平臺(tái)和linux平臺(tái)下都存在著大量的庫(kù)。
本質(zhì)上來(lái)說(shuō)庫(kù)是一種可執(zhí)行代碼的二進(jìn)制形式,可以被操作系統(tǒng)載入內(nèi)存執(zhí)行。
由于windows和linux的本質(zhì)不同,因此二者庫(kù)的二進(jìn)制是不兼容的。
2. 庫(kù)的種類
linux下的庫(kù)有兩種:靜態(tài)庫(kù)和共享庫(kù)(動(dòng)態(tài)庫(kù))。
二者的不同點(diǎn)在于代碼被載入的時(shí)刻不同。
靜態(tài)庫(kù)的代碼在編譯過(guò)程中已經(jīng)被載入可執(zhí)行程序,因此體積較大。共享庫(kù)的代碼是在可執(zhí)行程序運(yùn)行時(shí)才載入內(nèi)存的,在編譯過(guò)程中僅簡(jiǎn)單的引用,因此代碼體積較小。對(duì)于靜態(tài)庫(kù)和動(dòng)態(tài)庫(kù)的選擇,需要結(jié)合二者的優(yōu)缺點(diǎn)折中考慮。一般來(lái)說(shuō),比較通用的庫(kù),應(yīng)該做成共享庫(kù)。
3.庫(kù)存在的意義
庫(kù)是別人寫(xiě)好的現(xiàn)有的,成熟的,可以復(fù)用的代碼,你可以使用但要記得遵守許可協(xié)議。
現(xiàn)實(shí)中每個(gè)程序都要依賴很多基礎(chǔ)的底層庫(kù),不可能每個(gè)人的代碼都從零開(kāi)始,因此庫(kù)的存在意義非同尋常。共享庫(kù)的好處是,不同的應(yīng)用程序如果調(diào)用相同的庫(kù),那么 在內(nèi)存里只需要有一 份該共享庫(kù)的實(shí)例。
4. 庫(kù)文件在linux下是如何生成的
靜態(tài)庫(kù)的后綴是.a,它的產(chǎn)生分兩步
Step 1:由源文件編譯生成一堆.o,每個(gè).o里都包含這個(gè)編譯單元的符號(hào)表;
Step 2:ar命令將很多.o轉(zhuǎn)換成.a,成為靜態(tài)庫(kù);
動(dòng)態(tài)庫(kù)的后綴是.so,它由gcc加特定參數(shù)編譯產(chǎn)生。
例如: $ gcc -fPIC -c *.c $ gcc -shared -Wl,-soname, libfoo.so.1 -o libfoo.so.1.0 *.
5. 庫(kù)文件是如何命名的,有沒(méi)有什么規(guī)范
在linux下,庫(kù)文件一般放在/usr/lib和/lib下,
靜態(tài)庫(kù)的名字一般為libxxxx.a,其中xxxx是該lib的名稱
動(dòng)態(tài)庫(kù)的名字一般為libxxxx.so.major.minor,xxxx是該lib的名稱,major是主版本號(hào), minor是副版本號(hào)
6. 如何知道一個(gè)可執(zhí)行程序依賴哪些庫(kù)
ldd命令可以查看一個(gè)可執(zhí)行程序依賴的共享庫(kù),
例如# ldd /bin/lnlibc.so.6
=> /lib/libc.so.6 (0×40021000)/lib/ld-linux.so.2
=> /lib/ld- linux.so.2 (0×40000000)
可以看到ln命令依賴于libc庫(kù)和ld-linux庫(kù)
7. 可執(zhí)行程序在執(zhí)行的時(shí)候如何定位共享庫(kù)文件
靜態(tài)庫(kù):生成可執(zhí)行文件時(shí),靜態(tài)庫(kù)已經(jīng)作為自身一部分鏈接進(jìn)了可執(zhí)行文件中,故執(zhí)行時(shí)不需要再定位,也就是說(shuō)再不依賴于庫(kù)文件;
動(dòng)態(tài)庫(kù):需要知道動(dòng)態(tài)庫(kù)的路徑,參考另一篇博客;
8. 在新安裝一個(gè)庫(kù)之后如何讓系統(tǒng)能夠找到他
如果安裝在/lib或者/usr/lib下,那么ld默認(rèn)能夠找到,無(wú)需其他操作。
如果安裝在其他目錄,需要將其添加到/etc/ld.so.cache文件中,步驟如下:
1. 編輯/etc/ld.so.conf文件,加入庫(kù)文件所在目錄的路徑;
2. 運(yùn)行l(wèi)dconfig,該命令會(huì)重建/etc/ld.so.cache文件;
3. ldconfig命令需要root權(quán)限;
總結(jié)
以上就是本文關(guān)于Linux的庫(kù)文件的全部?jī)?nèi)容,希望對(duì)大家學(xué)習(xí)Linux有所幫助。歡迎大家參閱:淺談Linux環(huán)境下gcc優(yōu)化級(jí)別 詳解Docker使用Linux iptables 和 Interfaces管理容器網(wǎng)絡(luò)等。有什么問(wèn)題可以隨時(shí)留言,小編會(huì)及時(shí)回復(fù)大家的。
原文鏈接:http://www.linuxidc.com/Linux/2012-03/57023.htm