基礎知識來自 linux靜態庫和共用庫
編寫知識來自 linux中如何編譯靜態庫
1.什麼是庫
在windows平台和linux平台下都大量存在著庫。
本質上來說庫是一種可執行代碼的二進位形式,可以被作業系統載入記憶體執行。
由於windows和linux的本質不同,因此二者庫的二進位是不相容的。
本文僅限於介紹linux下的庫。
2.庫的種類
linux下的庫有兩種:靜態庫和共用庫(動態庫)。
二者的不同點在於代碼被載入的時刻不同。
靜態庫的代碼在編譯過程中已經被載入可執行程式,因此體積較大。
共用庫的代碼是在可執行程式運行時才載入記憶體的,在編譯過程中僅簡單的引用,因此代碼體積較小。
3.庫存在的意義
庫是別人寫好的現有的,成熟的,可以複用的代碼,你可以使用但要記得遵守許可協議。
現實中每個程式都要依賴很多基礎的底層庫,不可能每個人的代碼都從零開始,因此庫的存在意義非同尋常。
共用庫的好處是,不同的應用程式如果調用相同的庫,那麼在記憶體裡只需要有一份該共用庫的執行個體。
4.庫檔案是如何產生的在linux下
靜態庫的尾碼是.a,它的產生分兩步
Step 1.由源檔案編譯產生一堆.o,每個.o裡都包含這個編譯單元的符號表
Step 2.ar命令將很多.o轉換成.a,成文靜態庫
動態庫的尾碼是.so,它由gcc加特定參數編譯產生。
例如:
$ gcc -fPIC -c *.c $ gcc -shared -Wl,-soname, libfoo.so.1 -o libfoo.so.1.0 *.
5.庫檔案是如何命名的,有沒有什麼規範
在linux下,庫檔案一般放在/usr/lib /lib下,
靜態庫的名字一般為libxxxx.a,其中xxxx是該lib的名稱
動態庫的名字一般為libxxxx.so.major.minor,xxxx是該lib的名稱,major是主要版本號, minor是副版本號碼
6.如何知道一個可執行程式依賴哪些庫
ldd命令可以查看一個可執行程式依賴的共用庫,
例如# 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庫和ld-linux庫
7.可執行程式在執行的時候如何定位共用庫檔案
當系統載入可執行代碼時候,能夠知道其所依賴的庫的名字,但是還需要知道絕對路徑
此時就需要系統動態載入器(dynamic linker/loader)
對於elf格式的可執行程式,是由ld-linux.so*來完成的,它先後搜尋elf檔案的 DT_RPATH段—環境變數LD_LIBRARY_PATH—/etc/ld.so.cache檔案清單—/lib/,/usr/lib目錄找到庫檔案後將其載入記憶體
8.在新安裝一個庫之後如何讓系統能夠找到他
如果安裝在/lib或者/usr/lib下,那麼ld預設能夠找到,無需其他動作。
如果安裝在其他目錄,需要將其添加到/etc/ld.so.cache檔案中,步驟如下
1.編輯/etc/ld.so.conf檔案,加入庫檔案所在目錄的路徑
2.運行ldconfig,該命令會重建/etc/ld.so.cache檔案
erehw:
這樣的情況一般而言,很容易會碰到.我不知道csdn的faq是否已經有人整理過去.
一般而言,在C++原始碼中調用c的函數,如果對函數的申明沒有注意的話,就會出現這樣的問題
簡單的說
你應該這樣去申明一個c函數:
view plainprint?
- #ifdef __cplusplus
- #define EXTERN_C extern "C"
- #else
- #define EXTERN_C extern
- #endif
-
- EXTERN_C void test_9999();
#ifdef __cplusplus <br /> #define EXTERN_C extern "C" <br /> #else <br /> #define EXTERN_C extern <br /> #endif <br /> <br /> EXTERN_C void test_9999(); <br />
litw
如下使用應該沒有問題:
view plainprint?
- /* hellofirst.c */
- #include <stdio.h>
-
- void hellofirst()
- {
- printf("hello first! \n");
- }
-
- /* hellosecond.c */
- void hellosecond()
- {
- printf("hello second! \n");
- }
-
- $cc -c hellofirst.c hellosecond.c
- $ar -r libhello.a hellofirst.o hellosecond.o
-
- /* hello.c */
- void hellofirst(void);
- void hellosecond(void);
-
- int main(int argc, char *argv[])
- {
- hellofirst();
- hellosecond();
-
- return 0;
- }
/* hellofirst.c */ <br /> #include <stdio.h> <br /> <br /> void hellofirst() <br /> { <br /> printf("hello first! \n"); <br /> } <br /> <br /> /* hellosecond.c */ <br /> void hellosecond() <br /> { <br /> printf("hello second! \n"); <br /> } <br /> <br /> $cc -c hellofirst.c hellosecond.c <br /> $ar -r libhello.a hellofirst.o hellosecond.o <br /> <br /> /* hello.c */ <br /> void hellofirst(void); <br /> void hellosecond(void); <br /> <br /> int main(int argc, char *argv[]) <br /> { <br /> hellofirst(); <br /> hellosecond(); <br /> <br /> return 0; <br /> } <br />
引用 $cc hellotwice.c libhello.a -o hellotwice
在C++中C函數,如樓上所說,需要用extern "C"聲明