Linux靜態庫與動態庫實例詳解
1. Linux 下靜態鏈接庫編譯與使用
首先編寫如下代碼:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
|
// 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
|
1. gcc -o main main.o test .o |
1
2
3
4
5
6
7
8
|
【擴展】:其實上面為了讓大家更加清楚底層原因,我把編譯鏈接分開了,下面這樣編譯 也會報undefined reference錯,其實底層原因與上面是一樣的。 gcc -o main main.c // 缺少 test ()的實現文件 需要改成如下形式才能成功,將 test ()函數的實現文件一起編譯。 gcc -o main main.c test .c //ok ,沒問題了 |
4. 鏈接時缺少相關的庫文件(.a/.so)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
|
在此,只舉個靜態庫的例子,假設源碼與上述一致。 1. 把 test .c編譯成靜態庫: gcc -c test .c sr -rc test .a test .o gcc -c main.c 2. 生成可執行程序: gcc -o main -main.o 此時同樣出現 undefined reference to ` test '報錯。其根本原因也是找不到 test ()函數的實現文 件,由于該 test ()函數的實現在 test .a這個靜態庫中的,故在鏈接的時候需要在其后加入 test .a這個 庫,鏈接命令修改為如下形式即可。 1. gcc -o main main.c . /test .a |
5. 多個庫文件鏈接順序問題
1
2
3
4
5
6
7
8
9
10
11
12
|
這種問題也非常的隱蔽,不仔細研究你可能會感到非常地莫名其妙。我們依然回到第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的錯誤,完成編譯鏈接。 |
如有疑問請留言或者到本站社區交流討論,感謝閱讀,希望能幫助到大家,謝謝大家對本站的支持!
原文鏈接:http://blog.csdn.net/u013709270/article/details/53264141