静态库和动态库编译
靜態庫和動態庫的區別
庫從本質上來說是一種可執行代碼的二進制格式,可以被載入內存中執行。庫分靜態庫和動態庫兩種。?
1. 靜態函數庫
????這類庫的名字一般是libxxx.a;利用靜態函數庫編譯成的文件比較大,因為整個 函數庫的所有數據都會被整合進目標代碼中,他的優點就顯而易見了,即編譯后的執行程序不需要外部的函數庫支持,因為所有使用的函數都已經被編譯進去了。當然這也會成為他的缺點,因為如果靜態函數庫改變了,那么你的程序必須重新編譯。
2. 動態函數庫
????這類庫的名字一般是libxxx.so;相對于靜態函數庫,動態函數庫在編譯的時候 并沒有被編譯進目標代碼中,你的程序執行到相關函數時才調用該函數庫里的相應函數,因此動態函數庫所產生的可執行文件比較小。由于函數庫沒有被整合進你的程序,而是程序運行時動態的申請并調用,所以程序的運行環境中必須提供相應的庫。動態函數庫的改變并不影響你的程序,所以動態函數庫的升級比較方便。?
linux系統有幾個重要的目錄存放相應的函數庫,如/lib /usr/lib。
靜態庫的使用
靜態庫的操作工具:gcc和ar 命令。?
編寫及使用靜態庫?
(1)設計庫源碼 pr1.c 和 pr2.c?
[root@billstone make_lib]# cat pr1.c?void print1()? {?printf("This is the first lib src!\n");? }?[root@billstone make_lib]# cat pr2.c?void print2()? {?printf("This is the second src lib!\n");? }?(2)??編譯.c 文件?
[bill@billstone make_lib]$ cc -O -c pr1.c pr2.c?[bill@billstone make_lib]$ ls -l pr*.o?-rw-rw-r--????????1 bill??????????bill????????????????????804????4 月??15 11:11 pr1.o?-rw-rw-r--????????1 bill??????????bill????????????????????804????4 月??15 11:11 pr2.o?(3)??鏈接靜態庫?
????為了在編譯程序中正確找到庫文件,靜態庫必須按照 lib[name].a 的規則命名,如下例中[name]=pr.?
[bill@billstone make_lib]$ ar -rsv libpr.a pr1.o pr2.o?a - pr1.o? a - pr2.o?[bill@billstone make_lib]$ ls -l *.a?-rw-rw-r--????????1 bill??????????bill??????????????????1822????4 月??15 11:12 libpr.a?[bill@billstone make_lib]$ ar -t libpr.a?pr1.o? pr2.o?(4)??調用庫函數代碼 main.c?
[bill@billstone make_lib]$ cat main.c?int main()? {?print1();?print2();?return 0;? }?(5)??編譯鏈接選項?
????-L 及-l 參數放在后面.其中,-L 加載庫文件路徑,-l 指明庫文件名字.?
[bill@billstone make_lib]$ gcc -o main main.c -L./ -lpr?[bill@billstone make_lib]$ ls -l main*?-rwxrwxr-x????????1 bill??????????bill????????????????11805????4 月??15 11:17 main?-rw-rw-r--????????1 bill??????????bill??????????????????????50????4 月??15 11:15 main.c?(6)執行目標程序?
[bill@billstone make_lib]$ ./main?This is the first lib src!?This is the second src lib!?動態庫的使用
編寫動態庫?
(1)設計庫代碼?
[bill@billstone make_lib]$ cat pr1.c?int p = 2;?void print(){?printf("This is the first dll src!\n");?}?(2)生成動態庫?
[bill@billstone make_lib]$ gcc -O -fpic -shared -o libd1.so pr1.c?[bill@billstone make_lib]$ ls -l *.so?-rwxrwxr-x????????1 bill??????????bill??????????????????6592????4 月??15 15:19 libd1.so?(3)動態庫的隱式調用?
? ?在編譯調用庫函數代碼時指明動態庫的位置及名字,??看下面實例?
[bill@billstone make_lib]$ cat main.c?int main()? {?print();?return 0;? }?[bill@billstone make_lib]$ gcc -o tdl main.c ./libd1.so ?[bill@billstone make_lib]$ ./tdl?This is the first dll src!?? ?當動態庫的位置活名字發生改變時,??程序將無法正常運行;??而動態庫取代靜態庫的好處之一則是通過更新動態庫而隨時升級庫的內容.?
(4)態庫的動顯式調用?
? ? 顯式調用動態庫需要四個函數的支持,??函數 dlopen 打開動態庫,??函數 dlsym 獲取動態庫中對象基址,??函數 dlerror 獲取顯式動態庫操作中的錯誤信息,??函數 doclose 關閉動態庫.
[bill@billstone make_lib]$ cat main.c?#include <stdio.h> #include <stdlib.h>? int main()? {?void *pHandle;?void (*pFunc)();??????????????????????????????????//??指向函數的指針?int *p;?pHandle = dlopen("./libd1.so", RTLD_NOW);????????????//??在當前路徑(pwd)打開動態庫?if(!pHandle){?printf("Can't find libd1.so \n");?exit(1);?}?pFunc = (void (*)())dlsym(pHandle, "print");????? //??獲取庫函數 print 的地址?if(pFunc)?pFunc();?else?printf("Can't find function print\n");?p = (int *)dlsym(pHandle, "p");????????????????? //??獲取庫變量 p 的地址?if(p)?printf("p = %d\n", *p);?else?printf("Can't find int p\n");?dlclose(pHandle);????????????????????????? ???????//??關閉動態庫?return 0;? }?[bill@billstone make_lib]$?gcc -o tds main.c –libld1 –L. //編譯tds可執行文件, -L表示動態庫所在路徑,.表示當前路徑? ?此時還不能立即執行可執行文件tds,因為在動態函數庫使用時,會查找/usr/lib、/lib目錄下的動態函數庫,而此時我們生成的庫不在里邊。 這個時候有好幾種方法可以讓tds可執行文件成功運行:
①最直接最簡單的方法就是把libd1.so拉到/usr/lib或/lib中去;
② 還有一種方法執行命令#export LD_LIBRARY_PATH=$(pwd),將當前路徑加入到系統鏈接庫路徑中;
③另外還可以在/etc/ld.so.conf文件里加入我們生成的庫的目錄,然后執行命令#/sbin/ldconfig。 /etc/ld.so.conf是非常重要的一個目錄,里面存放的是鏈接器和加載器搜索共享庫時要檢查的目錄,默認是從/usr/lib或/lib中讀取的,所以想要順利運行,我們也可以把我們庫的目錄加入到這個文件中并直接執行命令#/sbin/ldconfig 。另外還有個文件需要了解/etc/ld.so.cache,里面保存了常用的動態函數庫,且會先把他們加載到內存中,因為內存的訪問速度遠遠大于硬盤的訪問速度,這樣可以提高軟件加載動態函數庫的速度了。
庫依賴的查看
使用ldd命令來查看執行文件依賴于哪些庫。
該命令用于判斷某個可執行的 binary 檔案含有什么動態函式庫。
[root@test root]# ldd [-vdr] [filename]
參數說明:
--version 打印ldd的版本號
-v --verbose 打印所有信息,例如包括符號的版本信息
-d --data-relocs 執行符號重部署,并報告缺少的目標對象(只對ELF格式適用)
-r --function-relocs 對目標對象和函數執行重新部署,并報告缺少的目標對象和函數(只對ELF格式適用)
--help 用法信息。
如果命令行中給定的庫名字包含'/',這個程序的libc5版本將使用它作為庫名字;否則它將在標準位置搜索庫。運行一個當前目錄下的共享庫,加前綴"./"
參考:http://blog.sina.com.cn/s/blog_744b57ce0102x6ow.html
總結
- 上一篇: 绝对强大的三个LINUX指令: AR,
- 下一篇: arm优化编译参数选项解释