linux動態編譯和靜態編譯
爲了使用方便,可以把這兩個函數接口定義爲動態鏈接庫或靜態鏈接庫。用動態鏈接庫編譯生成的
可執行文件需調用.so文件方可正常運行,靈活但稍顯麻煩;用靜態鏈接庫編譯生成的可執行文件可直
接運行,不用再調用如.so般的依賴庫文件,簡單但不靈活。
靜態鏈接庫:
1、編譯生成目標文件
gcc -c struct.c
2、創建靜態庫
ar cqs libstruct.a struct.o (順序不能亂)
3、鏈接靜態鏈接庫,生成可執行文件
gcc main.c -static -L. -lstruct -o main
動態鏈接庫:
1、編譯成動態鏈接庫
gcc struct.c -fPIC -shared -o libstruct.so
2、鏈接動態鏈接庫,生成可執行文件
gcc main.c -L. -lstruct -o main
3、設置庫文件的環境路徑
1)在bashrc或profile文件裏用LD_LIBRARY_PATH定義,然後用source加載。
2)把庫路徑添加到ld.so.conf文件中,然後用ldconfig加載。
3)ldconfig /home/user/lib,僅能暫時性使用,若下次ldconfig時此目錄下的動態鏈接庫就不能被共享
了。
gcc一些參數解析
-shared:指定生成動態鏈接庫。
-static:指定生成靜態鏈接庫。
-fPIC:表示編譯爲位置獨立的代碼,用於編譯共享庫。目標文件需要創建成位置無關碼,概念上就
是在可執行程序裝載它們的時候,它們可以放在可執行程序的內存裏的任何地方。
-L.:表示要連接的庫在當前目錄中。
-l:指定鏈接時需要的動態庫。編譯器查找動態連接庫時有隱含的命名規則,即在給出的名字前面加
上lib,後面加上.so來確定庫的名稱。
-Wall:生成所有警告信息。
-ggdb:此選項將盡可能的生成gdb的可以使用的調試信息。
-g:編譯器在編譯的時候產生調試信息。
-c:只激活預處理、編譯和彙編,也就是把程序做成目標文件(.o文件)。
-Wl,options:把參數(options)傳遞給鏈接器ld。如果options中間有逗號,就將options分成多個選項,然
後傳遞給鏈接程序。
摘要:本文主要通過三個程序來說明Linux動態庫和靜態庫的創建,以及使用它們。程序很簡單
但是足以反映出Linux動態庫和靜態庫的問題。
函數庫分爲靜態庫和動態庫兩種。創建Linux靜態庫和Linux動態庫和使用它們在這裏將以舉例的形式
詳述一下。靜態庫在程序編譯時會被連接到目標代碼中,程序運行時將不再需要該靜態庫。動態庫
在程序編譯時並不會被連接到目標代碼中,而是在程序運行是才被載入,因此在程序運行時還需要
動態庫存在。
第1步:編輯得到舉例的程序--hello.h、hello.c和main.c;
hello.h(見程序1)爲該函數庫的頭文件。
hello.c(見程序2)是函數庫的源程序,其中包含公用函數hello,該函數將在屏幕上輸出"Hello XXX!"。
main.c(見程序3)爲測試庫文件的主程序,在主程序中調用了公用函數hello。
--------------------------------------------------------------------------------
程序1: hello.h
#ifndef HELLO_H
#define HELLO_H
void hello(const char *name);
#endif //HELLO_H
--------------------------------------------------------------------------------
程序2: hello.c
#include <stdio.h>
void hello(const char *name)
{
printf("Hello %s!\n", name);
}
--------------------------------------------------------------------------------
程序3: main.c
#include "hello.h"
int main()
{
hello("everyone");
return 0;
}
--------------------------------------------------------------------------------
第2步:將hello.c編譯成.o文件
無論靜態庫,還是動態庫,都是由.o文件創建的。因此,我們必須將源程序hello.c通過gcc先編譯
成.o文件。在系統提示符下鍵入以下命令得到hello.o文件。
# gcc -c hello.c
第3步:由.o文件創建靜態庫
靜態庫文件名的命名規範是以lib爲前綴,緊接着跟靜態庫名,擴展名爲.a。例如:我們將創建的靜態
庫名爲myhello,則靜態庫文件名就是libmyhello.a。在創建和使用靜態庫時,需要注意這點。創建靜
態庫用ar命令。在系統提示符下鍵入以下命令將創建靜態庫文件libmyhello.a。
# ar cr libmyhello.a hello.o
第4步:在程序中使用靜態庫
靜態庫製作完了,如何使用它內部的函數呢?只需要在使用到這些公用函數的源程序中包含這些公
用函數的原型聲明,然後在用gcc命令生成目標文件時指明靜態庫名,gcc將會從靜態庫中將公用函
數連接到目標文件中。注意,gcc會在靜態庫名前加上前綴lib,然後追加擴展名.a得到的靜態庫文件
名來查找靜態庫文件。在程序3:main.c中,我們包含了靜態庫的頭文件hello.h,然後在主程序main
中直接調用公用函數hello。下面先生成目標程序hello,然後運行hello程序看看結果如何。
# gcc -o hello main.c -L. -lmyhello
# ./hello
Hello everyone!
#
我們刪除靜態庫文件試試公用函數hello是否真的連接到目標文件 hello中了。
# rm libmyhello.a
rm: remove regular file `libmyhello.a'? y
# ./hello
Hello everyone!
#
程序照常運行,靜態庫中的公用函數已經連接到目標文件中了。我們繼續看看如何在Linux中創建動
態庫。我們還是從.o文件開始。
第5步:由.o文件創建動態庫文件
動態庫文件名命名規範和靜態庫文件名命名規範類似,也是在動態庫名增加前綴lib,但其文件擴展
名爲.so。例如:我們將創建的動態庫名爲myhello,則動態庫文件名就是libmyhello.so。用gcc來創
建動態庫。在系統提示符下鍵入以下命令得到動態庫文件libmyhello.so。
# gcc -shared -fPCI -o libmyhello.so hello.o
第6步:在程序中使用動態庫
在程序中使用動態庫和使用靜態庫完全一樣,也是在使用到這些公用函數的源程序中包含這些公用
函數的原型聲明,然後在用gcc命令生成目標文件時指明動態庫名進行編譯。我們先運行gcc命令生
成目標文件,再運行它看看結果。
# gcc -o hello main.c -L. -lmyhello
# ./hello
./hello: error while loading shared libraries: libmyhello.so: cannot open shared object file: No such
file or directory
#
哦!出錯了。快看看錯誤提示,原來是找不到動態庫文件libmyhello.so。程序在運行時,會在/usr/lib
和/lib等目錄中查找需要的動態庫文件。若找到,則載入動態庫,否則將提示類似上述錯誤而終止程
序運行。我們將文件libmyhello.so複製到目錄/usr/lib中,再試試。
# mv libmyhello.so /usr/lib
# ./hello
Hello everyone!
#
成功了。這也進一步說明了動態庫在程序運行時是需要的。
我們回過頭看看,發現使用靜態庫和使用動態庫編譯成目標程序使用的gcc命令完全一樣,那當靜態
庫和動態庫同名時,gcc命令會使用哪個庫文件呢?抱着對問題必究到底的心情,來試試看。先刪除
除.c和.h外的 所有文件,恢復成我們剛剛編輯完舉例程序狀態。
# rm -f hello hello.o /usr/lib/libmyhello.so
# ls
hello.c hello.h main.c
#
在來創建靜態庫文件libmyhello.a和動態庫文件libmyhello.so。
# gcc -c hello.c
# ar cr libmyhello.a hello.o
# gcc -shared -fPCI -o libmyhello.so hello.o
# ls
hello.c hello.h hello.o libmyhello.a libmyhello.so main.c
#
通過上述最後一條ls命令,可以發現靜態庫文件libmyhello.a和動態庫文件libmyhello.so都已經生成,
並都在當前目錄中。然後,我們運行gcc命令來使用函數庫myhello生成目標文件hello,並運行程序
hello。
# gcc -o hello main.c -L. -lmyhello
# ./hello
./hello: error while loading shared libraries: libmyhello.so: cannot open shared object file: No such
file or directory
#
從程序hello運行的結果中很容易知道,當Linux靜態庫和Linux動態庫同名時, gcc命令將優先使用動
態庫。
可執行文件需調用.so文件方可正常運行,靈活但稍顯麻煩;用靜態鏈接庫編譯生成的可執行文件可直
接運行,不用再調用如.so般的依賴庫文件,簡單但不靈活。
靜態鏈接庫:
1、編譯生成目標文件
gcc -c struct.c
2、創建靜態庫
ar cqs libstruct.a struct.o (順序不能亂)
3、鏈接靜態鏈接庫,生成可執行文件
gcc main.c -static -L. -lstruct -o main
動態鏈接庫:
1、編譯成動態鏈接庫
gcc struct.c -fPIC -shared -o libstruct.so
2、鏈接動態鏈接庫,生成可執行文件
gcc main.c -L. -lstruct -o main
3、設置庫文件的環境路徑
1)在bashrc或profile文件裏用LD_LIBRARY_PATH定義,然後用source加載。
2)把庫路徑添加到ld.so.conf文件中,然後用ldconfig加載。
3)ldconfig /home/user/lib,僅能暫時性使用,若下次ldconfig時此目錄下的動態鏈接庫就不能被共享
了。
gcc一些參數解析
-shared:指定生成動態鏈接庫。
-static:指定生成靜態鏈接庫。
-fPIC:表示編譯爲位置獨立的代碼,用於編譯共享庫。目標文件需要創建成位置無關碼,概念上就
是在可執行程序裝載它們的時候,它們可以放在可執行程序的內存裏的任何地方。
-L.:表示要連接的庫在當前目錄中。
-l:指定鏈接時需要的動態庫。編譯器查找動態連接庫時有隱含的命名規則,即在給出的名字前面加
上lib,後面加上.so來確定庫的名稱。
-Wall:生成所有警告信息。
-ggdb:此選項將盡可能的生成gdb的可以使用的調試信息。
-g:編譯器在編譯的時候產生調試信息。
-c:只激活預處理、編譯和彙編,也就是把程序做成目標文件(.o文件)。
-Wl,options:把參數(options)傳遞給鏈接器ld。如果options中間有逗號,就將options分成多個選項,然
後傳遞給鏈接程序。
摘要:本文主要通過三個程序來說明Linux動態庫和靜態庫的創建,以及使用它們。程序很簡單
但是足以反映出Linux動態庫和靜態庫的問題。
函數庫分爲靜態庫和動態庫兩種。創建Linux靜態庫和Linux動態庫和使用它們在這裏將以舉例的形式
詳述一下。靜態庫在程序編譯時會被連接到目標代碼中,程序運行時將不再需要該靜態庫。動態庫
在程序編譯時並不會被連接到目標代碼中,而是在程序運行是才被載入,因此在程序運行時還需要
動態庫存在。
第1步:編輯得到舉例的程序--hello.h、hello.c和main.c;
hello.h(見程序1)爲該函數庫的頭文件。
hello.c(見程序2)是函數庫的源程序,其中包含公用函數hello,該函數將在屏幕上輸出"Hello XXX!"。
main.c(見程序3)爲測試庫文件的主程序,在主程序中調用了公用函數hello。
--------------------------------------------------------------------------------
程序1: hello.h
#ifndef HELLO_H
#define HELLO_H
void hello(const char *name);
#endif //HELLO_H
--------------------------------------------------------------------------------
程序2: hello.c
#include <stdio.h>
void hello(const char *name)
{
printf("Hello %s!\n", name);
}
--------------------------------------------------------------------------------
程序3: main.c
#include "hello.h"
int main()
{
hello("everyone");
return 0;
}
--------------------------------------------------------------------------------
第2步:將hello.c編譯成.o文件
無論靜態庫,還是動態庫,都是由.o文件創建的。因此,我們必須將源程序hello.c通過gcc先編譯
成.o文件。在系統提示符下鍵入以下命令得到hello.o文件。
# gcc -c hello.c
第3步:由.o文件創建靜態庫
靜態庫文件名的命名規範是以lib爲前綴,緊接着跟靜態庫名,擴展名爲.a。例如:我們將創建的靜態
庫名爲myhello,則靜態庫文件名就是libmyhello.a。在創建和使用靜態庫時,需要注意這點。創建靜
態庫用ar命令。在系統提示符下鍵入以下命令將創建靜態庫文件libmyhello.a。
# ar cr libmyhello.a hello.o
第4步:在程序中使用靜態庫
靜態庫製作完了,如何使用它內部的函數呢?只需要在使用到這些公用函數的源程序中包含這些公
用函數的原型聲明,然後在用gcc命令生成目標文件時指明靜態庫名,gcc將會從靜態庫中將公用函
數連接到目標文件中。注意,gcc會在靜態庫名前加上前綴lib,然後追加擴展名.a得到的靜態庫文件
名來查找靜態庫文件。在程序3:main.c中,我們包含了靜態庫的頭文件hello.h,然後在主程序main
中直接調用公用函數hello。下面先生成目標程序hello,然後運行hello程序看看結果如何。
# gcc -o hello main.c -L. -lmyhello
# ./hello
Hello everyone!
#
我們刪除靜態庫文件試試公用函數hello是否真的連接到目標文件 hello中了。
# rm libmyhello.a
rm: remove regular file `libmyhello.a'? y
# ./hello
Hello everyone!
#
程序照常運行,靜態庫中的公用函數已經連接到目標文件中了。我們繼續看看如何在Linux中創建動
態庫。我們還是從.o文件開始。
第5步:由.o文件創建動態庫文件
動態庫文件名命名規範和靜態庫文件名命名規範類似,也是在動態庫名增加前綴lib,但其文件擴展
名爲.so。例如:我們將創建的動態庫名爲myhello,則動態庫文件名就是libmyhello.so。用gcc來創
建動態庫。在系統提示符下鍵入以下命令得到動態庫文件libmyhello.so。
# gcc -shared -fPCI -o libmyhello.so hello.o
第6步:在程序中使用動態庫
在程序中使用動態庫和使用靜態庫完全一樣,也是在使用到這些公用函數的源程序中包含這些公用
函數的原型聲明,然後在用gcc命令生成目標文件時指明動態庫名進行編譯。我們先運行gcc命令生
成目標文件,再運行它看看結果。
# gcc -o hello main.c -L. -lmyhello
# ./hello
./hello: error while loading shared libraries: libmyhello.so: cannot open shared object file: No such
file or directory
#
哦!出錯了。快看看錯誤提示,原來是找不到動態庫文件libmyhello.so。程序在運行時,會在/usr/lib
和/lib等目錄中查找需要的動態庫文件。若找到,則載入動態庫,否則將提示類似上述錯誤而終止程
序運行。我們將文件libmyhello.so複製到目錄/usr/lib中,再試試。
# mv libmyhello.so /usr/lib
# ./hello
Hello everyone!
#
成功了。這也進一步說明了動態庫在程序運行時是需要的。
我們回過頭看看,發現使用靜態庫和使用動態庫編譯成目標程序使用的gcc命令完全一樣,那當靜態
庫和動態庫同名時,gcc命令會使用哪個庫文件呢?抱着對問題必究到底的心情,來試試看。先刪除
除.c和.h外的 所有文件,恢復成我們剛剛編輯完舉例程序狀態。
# rm -f hello hello.o /usr/lib/libmyhello.so
# ls
hello.c hello.h main.c
#
在來創建靜態庫文件libmyhello.a和動態庫文件libmyhello.so。
# gcc -c hello.c
# ar cr libmyhello.a hello.o
# gcc -shared -fPCI -o libmyhello.so hello.o
# ls
hello.c hello.h hello.o libmyhello.a libmyhello.so main.c
#
通過上述最後一條ls命令,可以發現靜態庫文件libmyhello.a和動態庫文件libmyhello.so都已經生成,
並都在當前目錄中。然後,我們運行gcc命令來使用函數庫myhello生成目標文件hello,並運行程序
hello。
# gcc -o hello main.c -L. -lmyhello
# ./hello
./hello: error while loading shared libraries: libmyhello.so: cannot open shared object file: No such
file or directory
#
從程序hello運行的結果中很容易知道,當Linux靜態庫和Linux動態庫同名時, gcc命令將優先使用動
態庫。
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.