Linux靜態庫與動態庫實例詳解
1. Linux 下靜態鏈接庫編譯與使用
首先編寫如下代碼:
// main.c#include "test.h"int main(){ test(); return 0; }// test.h#include<iostream>using namespace std;void test();// test.c#include "test.h"void test(){ cout<< "test!" <<endl;}
然后編譯:
1. gcc -c test.c //生成目標文件
2. ar crv libtest.a test.o //生成靜態鏈接庫libtest.a
3. g++ -o main main.c -ltest //編譯main程序同時鏈接libtest.a靜態庫
4. ./main //運行main程序
2. Linux 下動態鏈接庫編譯與使用
代碼與上述一致。
然后編譯:
1. g++ -fPIC -shared -o libtest.so test.c //生成動態鏈接庫libtest.so
2. g++ -o main main.c -ltest //調用動態鏈接庫libtest.so
3. ./main //運行main程序
3. 鏈接時缺失了相關目標文件(.o)
代碼與上述一致。
編譯過程如下:
1. gcc -c test.c
2. gcc -c main.c
3. gcc -o main main.o
這時,你會發現,報錯了:undefined reference to `test'.
這就是最典型的 undefined reference 錯誤,因為在鏈接時發現找不到某個函數的實現文件, 本例中test.o文件中包含了test()函數的實現,所以如果按下面這種方式鏈接就沒事了。
1. gcc -o main main.o test.o
【擴展】:其實上面為了讓大家更加清楚底層原因,我把編譯鏈接分開了,下面這樣編譯也會報undefined reference錯,其實底層原因與上面是一樣的。gcc -o main main.c //缺少test()的實現文件 需要改成如下形式才能成功,將test()函數的實現文件一起編譯。gcc -o main main.c test.c //ok,沒問題了
4. 鏈接時缺少相關的庫文件(.a/.so)
在此,只舉個靜態庫的例子,假設源碼與上述一致。1. 把test.c編譯成靜態庫: gcc -c test.c sr -rc test.a test.o gcc -c main.c2. 生成可執行程序: gcc -o main -main.o 此時同樣出現 undefined reference to `test'報錯。其根本原因也是找不到test()函數的實現文 件,由于該test()函數的實現在test.a這個靜態庫中的,故在鏈接的時候需要在其后加入test.a這個 庫,鏈接命令修改為如下形式即可。 1. gcc -o main main.c ./test.a
5. 多個庫文件鏈接順序問題
這種問題也非常的隱蔽,不仔細研究你可能會感到非常地莫名其妙。我們依然回到第3小節所討論的問題中,在最后,如果我們把鏈接的庫的順序換一下,看看會發生什么結果?1. gcc -o main main.o func.a test.a我們會得到如下的編譯錯誤:1. test.a(test.o): In function `test': 2. test.c:(.text+0x13): undefined reference to `func' 3. collect2: ld returned 1 exit status 因此,我們需要注意,在鏈接命令中給出所依賴的庫時,需要注意庫之間的依賴順序,依賴其他庫的庫一定要放到被依賴庫的前面,這樣才能真正避免undefined reference的錯誤,完成編譯鏈接。
如有疑問請留言或者到本站社區交流討論,感謝閱讀,希望能幫助到大家,謝謝大家對本站的支持!
新聞熱點
疑難解答