CC++編譯器和調試器以及靜態(tài)庫、動態(tài)庫使用匯總_第1頁
CC++編譯器和調試器以及靜態(tài)庫、動態(tài)庫使用匯總_第2頁
CC++編譯器和調試器以及靜態(tài)庫、動態(tài)庫使用匯總_第3頁
CC++編譯器和調試器以及靜態(tài)庫、動態(tài)庫使用匯總_第4頁
CC++編譯器和調試器以及靜態(tài)庫、動態(tài)庫使用匯總_第5頁
已閱讀5頁,還剩10頁未讀, 繼續(xù)免費閱讀

下載本文檔

版權說明:本文檔由用戶提供并上傳,收益歸屬內容提供方,若內容存在侵權,請進行舉報或認領

文檔簡介

1、C/C+ 編譯器和調試器以及靜態(tài)庫、動態(tài)庫使用匯總閱讀提示:本文是C/C+ 編譯器和調試器以及靜態(tài)庫、動態(tài)庫使用匯總 大多數unix系統(tǒng)下面的調試器的使用方法如下: gdb介紹GNU 的調試器稱為 gdb,該程序是一個交互式工具,工作在字符模式。在 X Window 系統(tǒng)中,有一個 gdb 的 前端圖形工具,稱為 xxgdb。gdb 是功能強大的調試程序,可完成如下的調試任務: * 設置斷點; * 監(jiān)視程序變量的值; * 程序的單步執(zhí)行; * 修改變量的值。 在可以使用 gdb 調試程序之前,必須使用 -g 選項編譯源文件??稍?makefile 中如下定義 CFLAGS 變量: CFLAGS

2、 = -g 運行 gdb 調試程序時通常使用如下的命令: gdb progname 在 gdb 提示符處鍵入help,將列出命令的分類,主要的分類有: * aliases:命令別名 * breakpoints:斷點定義; * data:數據查看; * files:指定并查看文件; * internals:維護命令; * running:程序執(zhí)行; * stack:調用棧查看; * statu:狀態(tài)查看; * tracepoints:跟蹤程序執(zhí)行。 鍵入 help 后跟命令的分類名,可獲得該類命令的詳細清單。 gdb 的常用命令命令 解釋 break NUM 在指定的行上設置斷點。 bt 顯示所

3、有的調用棧幀。該命令可用來顯示函數的調用順序。 clear 刪除設置在特定源文件、特定行上的斷點。其用法為:clear FILENAME:NUM。 continue 繼續(xù)執(zhí)行正在調試的程序。該命令用在程序由于處理信號或斷點而 導致停止運行時。 display EXPR 每次程序停止后顯示表達式的值。表達式由程序定義的變量組成。 file FILE 裝載指定的可執(zhí)行文件進行調試。 help NAME 顯示指定命令的幫助信息。 info break 顯示當前斷點清單,包括到達斷點處的次數等。 info files 顯示被調試文件的詳細信息。 info func 顯示所有的函數名稱。 info lo

4、cal 顯示當函數中的局部變量信息。 info prog 顯示被調試程序的執(zhí)行狀態(tài)。 info var 顯示所有的全局和靜態(tài)變量名稱。 kill 終止正被調試的程序。 list 顯示源代碼段。 make 在不退出 gdb 的情況下運行 make 工具。 next 在不單步執(zhí)行進入其他函數的情況下,向前執(zhí)行一行源代碼。 print EXPR 顯示表達式 EXPR 的值。 gdb 使用范例- 清單 一個有錯誤的 C 源程序 bugging.c - #include #include static char buff 256; static char* string; int main () pri

5、ntf ("Please input a string: "); gets (string); printf ("nYour string is: %sn", string); - 上面這個程序非常簡單,其目的是接受用戶的輸入,然后將用戶的輸入打印出來。該程序使用了一個未經過初 始化的字符串地址 string,因此,編譯并運行之后,將出現(xiàn) Segment Fault 錯誤: $ gcc -o test -g test.c $ ./test Please input a string: asfd Segmentation fault (core dumpe

6、d) 為了查找該程序中出現(xiàn)的問題,我們利用 gdb,并按如下的步驟進行: 1運行 gdb bugging 命令,裝入 bugging 可執(zhí)行文件; 2執(zhí)行裝入的 bugging 命令; 3使用 where 命令查看程序出錯的地方; 4利用 list 命令查看調用 gets 函數附近的代碼; 5唯一能夠導致 gets 函數出錯的因素就是變量 string。用 print 命令查看 string 的值; 6在 gdb 中,我們可以直接修改變量的值,只要將 string 取一個合法的指針值就可以了,為此,我們在第 11 行處設置斷點; 7程序重新運行到第 11 行處停止,這時,我們可以用 set v

7、ariable 命令修改 string 的取值; 8然后繼續(xù)運行,將看到正確的程序運行結果。 運行 gcc/egcs GCC 是 GNU 的 C 和 C+ 編譯器。實際上,GCC 能夠編譯三種語言:C、C+ 和 Object C(C 語言的一種面向對象擴展)。利用 gcc 命令可同時編譯并連接 C 和 C+ 源程序。 如果你有兩個或少數幾個 C 源文件,也可以方便地利用 GCC 編譯、連接并生成可執(zhí)行文件。例如,假設你有 兩個源文件 main.c 和 factorial.c 兩個源文件,現(xiàn)在要編譯生成一個計算階乘的程序。 清單 factorial.c - #include #include i

8、nt factorial (int n) if (n <= 1) return 1; else return factorial (n - 1) * n; - - 清單 main.c - #include #include int factorial (int n); int main (int argc, char *argv) int n; if (argc < 2) printf ("Usage: %s nn", argv 0); return -1; else n = atoi (argv1); printf ("Factorial of %d

9、 is %d.n", n, factorial (n); return 0; - 利用如下的命令可編譯生成可執(zhí)行文件,并執(zhí)行程序: $ gcc -o factorial main.c factorial.c $ ./factorial 5 Factorial of 5 is 120. GCC 可同時用來編譯 C 程序和 C+ 程序。一般來說,C 編譯器通過源文件的后綴名來判斷是 C 程序還是 C+ + 程序。在 Linux 中,C 源文件的后綴名為 .c,而 C+ 源文件的后綴名為 .C 或 .cpp。 但是,gcc 命令只能編譯 C+ 源文件,而不能自動和 C+ 程序使用的庫連接。

10、因此,通常使用 g+ 命令來完 完成 C+ 程序的編譯和連接,該程序會自動調用 gcc 實現(xiàn)編譯。 假設我們有一個如下的 C+ 源文件(hello.C): #include void main (void) cout << "Hello, world!" << endl; 則可以如下調用 g+ 命令編譯、連接并生成可執(zhí)行文件: $ g+ -o hello hello.C $ ./hello Hello, world! gcc/egcs 的主要選項選項 解釋 -ansi 只支持 ANSI 標準的 C 語法。這一選項將禁止 GNU C 的某些特色, 例如

11、 asm 或 typeof 關鍵詞。 -c 只編譯并生成目標文件。 -DMACRO 以字符串“1”定義 MACRO 宏。 -DMACRO=DEFN 以字符串“DEFN”定義 MACRO 宏。 -E 只運行 C 預編譯器。 -g 生成調試信息。GNU 調試器可利用該信息。 -IDIRECTORY 指定額外的頭文件搜索路徑DIRECTORY。 -LDIRECTORY 指定額外的函數庫搜索路徑DIRECTORY。 -lLIBRARY 連接時搜索指定的函數庫LIBRARY。 -m486 針對 486 進行代碼優(yōu)化。 -o FILE 生成指定的輸出文件。用在生成可執(zhí)行文件時。 -O0 不進行優(yōu)化處理。

12、-O 或 -O1 優(yōu)化生成代碼。 -O2 進一步優(yōu)化。 -O3 比 -O2 更進一步優(yōu)化,包括 inline 函數。 -shared 生成共享目標文件。通常用在建立共享庫時。 -static 禁止使用共享連接。 -UMACRO 取消對 MACRO 宏的定義。 -w 不生成任何警告信息。 -Wall 生成所有警告信息。 SCO UNIX下面dbaxtra的調試技術 在sco unix下編程大多離不開C語言,即使是數據庫應用也有很多是與c搭配使用的,例如informix esql/c 就可以在c語言中嵌入sql 語句。很多人認為在unix下寫程序是件很痛苦的事情,其中一個很重要原因是不知

13、道在unix下怎樣調試程序。其實在sco unix源碼調試器是dbxtra或dbXtra,linux下是gdb。它們類似turbo c的調試器,可以跟蹤源碼變量。在unix 下調試程序有如下傳統(tǒng)方法 一、在要調試語句之前,輸出要調試的變量,利用printf()函數。 二、寫日志文件,把結果輸出到文件中避免屏幕混亂,利用fprintf()函數。 三、利用sco 內置調試器dbxtra或dbXtra。 dbxtra 適用字符界面,在sco unix的圖形界面用dbXtra。(編按:請注意大小寫) 以下是dbxtra基本命令: c cont 在斷點后繼續(xù)執(zhí)行 d delete 刪除所設斷點 h he

14、lp 幫助 e edit 編輯源程序 n next 源程序區(qū)的內容向下翻一屏。 p print 顯示變量 q quit 退出dbxtra r run 運行程序,直到遇上設置的斷點 rr rerun 再次運行 s step 單步運行 st stop 設置斷點 j status 顯示當前斷點 t where 顯示當前狀態(tài),列出所有設置的變量值 di display 開顯示窗,用于查看變量 ud undisplay 刪除顯示窗的條目 f forward 源程序區(qū)的內容向上 翻一屏。 B backward 源程序區(qū)的內容向下 翻一屏。 Stopi stop inst 設置斷點 tracei trace

15、 inst跟蹤子程序 dbxtra options objectfile dbxtra 在啟動時有個參數-Idir值得一提.我們在編寫一個較大程序的時候,通常源程序和編譯生成的可執(zhí)行文件都放在不同的目錄中,這樣便于管理。默認dbxtra將在可執(zhí)行文件所在的目錄下找匹配c的源程序。當我們啟動時,指定-I參數,dbxtra就會到我們指定的目錄下找匹配的c程序。 例如: dbxtra -I"workc" program1 源程序在用cc編譯時要帶上-g 參數,這樣是加上符號表等調試信息。只有這樣編譯過的文件,dbxtra才可以調試。調試信息使源代碼和機器碼關聯(lián)。 下面這個C程序輸

16、出結果和我們的預想結果不一樣,說明某些地方有錯誤。我們用調試器來調試它: 程序一: t.c main() int i=10 ,*p1; float j=1.5,*p2; p1=& p2=& p2=p1; printf("%d,%dn",*p1,*p2); 首先帶上-g參數編譯 cc -g -o t t.c 啟動調試器 dbxtra t 屏幕顯示: 1.main() 2. int i=10 ,*p1; 3. float j=1.5,*p2; 4. p1=& 5. p2=& 6. p2=p1; 7. printf("%d,%dn&quo

17、t;,*p1,*p2); 8. Cbrowse File:t.c Func.- Readubg symbolic information Type 'help' for help (dbxtra) (dbxtra) 設置斷點: (dbxtra)stop at 5 運行: (dbxtra) run 程序自動在第5行停下。 這時我們可以看變量的值。 (dbxtra) print *p1 單步執(zhí)行。 (dbxtra) step 程序將執(zhí)行第5行源碼,指針將移到第6行。 (dbxtra) print *p2 (dbxtra) step 程序執(zhí)行了第6行源碼后,將指針移到第7行。 (db

18、xtra) print *p1 , *p2 我們發(fā)現(xiàn) 在執(zhí)行了第6行源碼后,*p1,*p2的值就不對了,所以問題就出在第6行上。仔細檢查后發(fā)現(xiàn)指針p1指向整型,指針p2指向實型。它們之間的賦值要進行強制類型轉換。這種錯誤在C程序中是很常見的。 有時我們在調試一些程序時,要在整個程序運行中時刻監(jiān)視莫些變量的值,例如程序一中我們要時刻了解*p1,*p2的值,除了在每一行程序執(zhí)行完后,打print *p1,*p2外,還可以開一個顯示窗口。 (dbxtra)display *p1,*p2 用undisplay 刪掉不想要的變量。 有些程序運行時要帶參數,mycat /etc/passwd 在調試時候

19、(dbxtra) run '/etc/passwd' 再運行時,無需再寫一遍參數。 (dbxtra) rerun 在涉及到curses庫編程或屏幕有大量的人機界面時,為了調試方便,我們可以把程序輸出結果重定向到個虛屏。 (dbxtra) run >/dev/tty03 當然要先把tty03 disable 掉。(disable tty03) 創(chuàng)建和使用靜態(tài)庫詳細的使用情況,請大家man手冊,這里只介紹一下。靜態(tài)庫相對的比較簡單。 創(chuàng)建一個靜態(tài)庫是相當簡單的。通常使用 ar 程序把一些目標文件(.o)組合在一起,成為一個單獨的庫,然后運行 ranlib,以給庫加入一些索引信

20、息。 創(chuàng)建和使用共享庫特殊的編譯和連接選項 -D_REENTRANT 使得預處理器符號 _REENTRANT 被定義,這個符號激活一些宏特性。 -fPIC 選項產生位置獨立的代碼。由于庫是在運行的時候被調入,因此這個選項是必需的,因為在編譯的時候,裝入內存的地址還不知道。如果不使用這個選項,庫文件可能不會正確運行。 -shared 選項告訴編譯器產生共享庫代碼。 -Wl,-soname -Wl 告訴編譯器將后面的參數傳遞到連接器。而 -soname 指定了共享庫的 soname。 可以把庫文件拷貝到 /etc/ld.so.conf 中列舉出的任何目錄中,并以 root 身份運行 ldconfi

21、g;或者 運行 export LD_LIBRARY_PATH='pwd',它把當前路徑加到庫搜索路徑中去。 使用高級共享庫特性1. ldd 工具 ldd 用來顯示執(zhí)行文件需要哪些共享庫, 共享庫裝載管理器在哪里找到了需要的共享庫. 2. soname 共享庫的一個非常重要的,也是非常難的概念是 soname簡寫共享目標名(short for shared object name)。這是一個為共享庫(.so)文件而內嵌在控制數據中的名字。如前面提到的,每一個程序都有一個需要使用的庫的清單。這個清單的內容是一系列庫的 soname,如同 ldd 顯示的那樣,共享庫裝載器必須找到這個

22、清單。 soname 的關鍵功能是它提供了兼容性的標準。當要升級系統(tǒng)中的一個庫時,并且新庫的 soname 和老的庫的 soname 一樣,用舊庫連接生成的程序,使用新的庫依然能正常運行。這個特性使得在 Linux 下,升級使用共享庫的程序和定位錯誤變得十分容易。 在 Linux 中,應用程序通過使用 soname,來指定所希望庫的版本。庫作者也可以通過保留或者改變 soname 來聲明,哪些版本是相互兼容的,這使得程序員擺脫了共享庫版本沖突問題的困擾。 查看/usr/local/lib 目錄,分析 MiniGUI 的共享庫文件之間的關系 3. 共享庫裝載器 當程序被調用的時候,Linux 共

23、享庫裝載器(也被稱為動態(tài)連接器)也自動被調用。它的作用是保證程序所需要的所有適當版本的庫都被調入內存。共享庫裝載器名字是 ld.so 或者是 ld-linux.so,這取決于 Linux libc 的版本,它必須使用一點外部交互,才能完成自己的工作。然而它接受在環(huán)境變量和配置文件中的配置信息。 文件 /etc/ld.so.conf 定義了標準系統(tǒng)庫的路徑。共享庫裝載器把它作為搜索路徑。為了改變這個設置,必須以 root 身份運行 ldconfig 工具。這將更新 /etc/ls.so.cache 文件,這個文件其實是裝載器內部使用的文件之一。 可以使用許多環(huán)境變量控制共享庫裝載器的操作(表1-

24、4+)。 表 1-4+ 共享庫裝載器環(huán)境變量 變量 含義 LD_AOUT_LIBRARY_PATH 除了不使用 a.out 二進制格式外,與 LD_LIBRARY_PATH 相同。 LD_AOUT_PRELOAD 除了不使用 a.out 二進制格式外,與 LD_PRELOAD 相同。 LD_KEEPDIR 只適用于 a.out 庫;忽略由它們指定的目錄。 LD_LIBRARY_PATH 將其他目錄加入庫搜索路徑。它的內容應該是由冒號 分隔的目錄列表,與可執(zhí)行文件的 PATH 變量具有相同的格式。 如果調用設置用戶 ID 或者進程 ID 的程序,該變量被忽略。 LD_NOWARN 只適用于 a.

25、out 庫;當改變版本號是,發(fā)出警告信息。 LD_PRELOAD 首先裝入用戶定義的庫,使得它們有機會覆蓋或者重新定義標準庫。 使用空格分開多個入口。對于設置用戶 ID 或者進程 ID 的程序, 只有被標記過的庫才被首先裝入。在 /etc/ld.so.perload 中指定 了全局版本號,該文件不遵守這個限制。 4. 使用 dlopen 另外一個強大的庫函數是 dlopen()。該函數將打開一個新庫,并把它裝入內存。該函數主要用來加載庫中的符號,這些符號在編譯的時候是不知道的。比如 Apache Web 服務器利用這個函數在運行過程中加載模塊,這為它提供了額外的能力。一個配置文件控制了加載模塊

26、的過程。這種機制使得在系統(tǒng)中添加或者刪除一個模塊時,都不需要重新編譯了。 可以在自己的程序中使用 dlopen()。dlopen() 在 dlfcn.h 中定義,并在 dl 庫中實現(xiàn)。它需要兩個參數:一個文件名和一個標志。文件名可以是我們學習過的庫中的 soname。標志指明是否立刻計算庫的依賴性。如果設置為 RTLD_NOW 的話,則立刻計算;如果設置的是 RTLD_LAZY,則在需要的時候才計算。另外,可以指定 RTLD_GLOBAL,它使得那些在以后才加載的庫可以獲得其中的符號。 當庫被裝入后,可以把 dlopen() 返回的句柄作為給 dlsym() 的第一個參數,以獲得符號在庫中的地

27、址。使用這個地址,就可以獲得庫中特定函數的指針,并且調用裝載庫中的相應函數。LINUX動態(tài)鏈接庫的使用 一、編寫合格的動態(tài)鏈接庫頭文件 C語言的頭文件,可供一個或多個程序引用,里面一般定義程序所需的常量,自定義類型及函數原型說明等.其中的函數原型說明,則供編譯器檢查語法,用于排除引用參數時類型不一致的錯誤.只有編寫合格的動態(tài)鏈接庫頭文件,程序員才能正確使用動態(tài)鏈接庫內的函數. 動態(tài)鏈接庫頭文件要采用C語言標準格式,其中的動態(tài)函數原型定義,不必象上文介紹的那樣用(*動態(tài)函數名)的描述形式.請看下面的例子每行開始的數字為所在行行號,為筆者添加,供注解使用) 1 /* adatetime.h : 縱

28、橫軟件制作中心雨亦奇(zhsoft)編寫, 2002-03-06. */ 2 3 #ifndef _DATETIME_H 4 5 #define _DATETIME_H 6 7 /* 日期結構 */ 8 typedef struct 9 10 int year; 11 int mon; 12 int day; 13 DATETYPE; 14 15 /* 時間結構 */ 16 typedef struct 17 18 char hour; 19 char min; 20 char sec; 21 TIMETYPE; 22 23 int getdate(DATETYPE *d); /* 取當前日期

29、 */ 24 int gettime(TIMETYPE *t); /* 取當前時間 */ 25 26 #endif 27 注:與上文的datetime.h文件比較,從該頭文件第23,24行可以看到,動態(tài)函數getdate,gettime的原型定義改變了,不再使用(*getdate),(*gettime)的格式了(這種格式使用較為羅嗦). 二、正確編譯與命名動態(tài)鏈接庫 為了讓GCC編譯器生成動態(tài)鏈接庫,編譯時須加選項-shared.(這點須牢記) LINUX系統(tǒng)中,為了讓動態(tài)鏈接庫能被系統(tǒng)中其它程序共享,其名字應符合“l(fā)ib*.so*”這種格式.如果某個動態(tài)鏈接庫不符合此格式,則LINUX的動態(tài)

30、鏈接庫自動裝入程序(ld.so)將搜索不到此鏈接庫,其它程序也無法共享之. 格式中,第一個*通常表示為簡寫的庫名,第二個*通常表示為該庫的版本號.如:在我的系統(tǒng)中,基本C動態(tài)鏈接庫的名字為libc.so.6,線程pthread動態(tài)鏈接庫的名字為libpthread.so.0等等.本文例子所生成的動態(tài)鏈接庫的名字為libmy.so,雖沒有版本號,但也符合所要求的格式. 生成該動態(tài)鏈接庫的維護文件makefile-lib內容如下: 1 # makefile : 縱橫軟件制作中心雨亦奇編寫, 2002-03-07. 2 3 all : libmy.so 4 5 SRC = getdate.c get

31、time.c 6 7 TGT = $(SRC:.c=.o) 8 9 $(SRC) : adatetime.h 10 touch $ 11 12 %.o : %.c 13 cc -c $? 14 15 # 動態(tài)鏈接庫(libmy.so)生成 16 libmy.so : $(TGT) 17 cc -s -shared -o $ $(TGT) 18 運行命令: $ make -f makefile-lib $ 即生成libmy.so庫. 注: 維護文件中,第17行用-shared選項以生成動態(tài)鏈接庫,用-s選項以去掉目標文件中的符號表,從而減小文件長度. 三、共享動態(tài)鏈接庫 3.1 動態(tài)鏈接庫配置文

32、件 為了讓動態(tài)鏈接庫為系統(tǒng)所使用,需要維護動態(tài)鏈接庫的配置文件-/etc/ld.so.conf.此文件內,存放著可被LINUX共享的動態(tài)鏈接庫所在目錄的名字(系統(tǒng)目錄/lib,/usr/lib除外),各個目錄名間以空白字符(空格,換行等)或冒號或逗號分隔.一般的LINUX發(fā)行版中,此文件均含一個共享目錄/usr/X11R6/lib,為X window窗口系統(tǒng)的動態(tài)鏈接庫所在的目錄. 下面看看我的系統(tǒng)中此文件的內容如何: # cat /etc/ld.so.conf /usr/X11R6/lib /usr/zzz/lib # 由上可以看出,該動態(tài)庫配置文件中,增加了一個/usr/zzz/lib目錄

33、.這是我自己新建的共享庫目錄,下面存放我新開發(fā)的可供系統(tǒng)共享的動態(tài)鏈接庫. 3.2 動態(tài)鏈接庫管理命令 為了讓動態(tài)鏈接庫為系統(tǒng)所共享,還需運行動態(tài)鏈接庫的管理命令-ldconfig.此執(zhí)行程序存放在/sbin目錄下. ldconfig命令的用途,主要是在默認搜尋目錄(/lib和/usr/lib)以及動態(tài)庫配置文件/etc/ld.so.conf內所列的目錄下,搜索出可共享的動態(tài)鏈接庫(格式如前介紹,lib*.so*),進而創(chuàng)建出動態(tài)裝入程序(ld.so)所需的連接和緩存文件.緩存文件默認為/etc/ld.so.cache,此文件保存已排好序的動態(tài)鏈接庫名字列表. ldconfig通常在系統(tǒng)啟動時

34、運行,而當用戶安裝了一個新的動態(tài)鏈接庫時,就需要手工運行這個命令. ldconfig命令行用法如下: ldconfig -v|-verbose -n -N -X -f CONF -C CACHE -r ROOT -l -p|-print-cache -c FORMAT -format=FORMAT -V -?|-help|-usage path. ldconfig可用的選項說明如下: (1) -v或-verbose : 用此選項時,ldconfig將顯示正在掃描的目錄及搜索到的動態(tài)鏈接庫,還有它所創(chuàng)建的連接的名字. (2) -n : 用此選項時,ldconfig僅掃描命令行指定的目錄,不掃描默

35、認目錄(/lib,/usr/lib),也不掃描配置文件/etc/ld.so.conf所列的目錄. (3) -N : 此選項指示ldconfig不重建緩存文件(/etc/ld.so.cache).若未用-X選項,ldconfig照常更新文件的連接. (4) -X : 此選項指示ldconfig不更新文件的連接.若未用-N選項,則緩存文件正常更新. (5) -f CONF : 此選項指定動態(tài)鏈接庫的配置文件為CONF,系統(tǒng)默認為/etc/ld.so.conf. (6) -C CACHE : 此選項指定生成的緩存文件為CACHE,系統(tǒng)默認的是/etc/ld.so.cache,此文件存放已排好序的可共

36、享的動態(tài)鏈接庫的列表. (7) -r ROOT : 此選項改變應用程序的根目錄為ROOT(是調用chroot函數實現(xiàn)的).選擇此項時,系統(tǒng)默認的配置文件/etc/ld.so.conf,實際對應的為ROOT/etc/ld.so.conf.如用-r /usr/zzz時,打開配置文件/etc/ld.so.conf時,實際打開的是/usr/zzz/etc/ld.so.conf文件.用此選項,可以大大增加動態(tài)鏈接庫管理的靈活性. 通常情況下,ldconfig搜索動態(tài)鏈接庫時將自動建立動態(tài)鏈接庫的連接.選擇此項時,將進入專家模式,需要手工設置連接.一般用戶不用此項. (9) -p或-print-cache

37、 : 此選項指示ldconfig打印出當前緩存文件所保存的所有共享庫的名字. (10) -c FORMAT 或 -format=FORMAT : 此選項用于指定緩存文件所使用的格式,共有三種:old(老格式),new(新格式)和compat(兼容格式,此為默認格式). (11) -V : 此選項打印出ldconfig的版本信息,而后退出. (12) -? 或 -help 或 -usage : 這三個選項作用相同,都是讓ldconfig打印出其幫助信息,而后退出. 舉三個例子: 例1: # ldconfig -p 793 libs found in cache /etc/ld.so.cache&

38、#39; libzvt.so.2 (libc6) => /usr/lib/libzvt.so.2 libzvt.so (libc6) => /usr/lib/libzvt.so libz.so.1.1.3 (libc6) => /usr/lib/libz.so.1.1.3 libz.so.1 (libc6) => /lib/libz.so.1 . # 注: 有時候用戶想知道系統(tǒng)中有哪些動態(tài)鏈接庫,或者想知道系統(tǒng)中有沒有某個動態(tài)鏈接庫,這時,可用-p選項讓ldconfig輸出緩存文件中的動態(tài)鏈接庫列表,從而查詢得到.例子中,ldconfig命令的輸出結果第1行表明在緩存文

39、件/etc/ld.so.cache中找到793個共享庫,第2行開始便是一系列共享庫的名字及其全名(絕對路徑).因為實際輸出結果太多,為節(jié)省篇幅,以.表示省略的部分. 例2: # ldconfig -v /lib: liby.so.1 -> liby.so.1 libnss_wins.so -> libnss_wins.so . /usr/lib: libjscript.so.2 -> libjscript.so.2.0.0 libkspell.so.2 -> libkspell.so.2.0.0 . /usr/X11R6/lib: libmej-0.8.10.so -&

40、gt; libmej-0.8.10.so libXaw3d.so.7 -> libXaw3d.so.7.0 . # 注: ldconfig命令在運行正常的情況下,默認不輸出什么東西.本例中用了-v選項,以使ldconfig在運行時輸出正在掃描的目錄及搜索到的共享庫,用戶可以清楚地看到運行的結果.執(zhí)行結束后,ldconfig將刷新緩存文件/etc/ld.so.cache. 例3: # ldconfig /usr/zhsoft/lib # 注: 當用戶在某個目錄下面創(chuàng)建或拷貝了一個動態(tài)鏈接庫,若想使其被系統(tǒng)共享,可以執(zhí)行一下"ldconfig 目錄名"這個命令.此命令的功

41、能在于讓ldconfig將指定目錄下的動態(tài)鏈接庫被系統(tǒng)共享起來,意即:在緩存文件/etc/ld.so.cache中追加進指定目錄下的共享庫.本例讓系統(tǒng)共享了/usr/zhsoft/lib目錄下的動態(tài)鏈接庫.需要說明的是,如果此目錄不在/lib,/usr/lib及/etc/ld.so.conf文件所列的目錄里面,則再度運行l(wèi)dconfig時,此目錄下的動態(tài)鏈接庫可能不被系統(tǒng)共享了. 3.3 動態(tài)鏈接庫如何共享 了解了以上知識,我們可以采用以下三種方法來共享動態(tài)鏈接庫注:均須在超級用戶狀態(tài)下操作,以我的動態(tài)鏈接庫libmy.so共享過程為例) (1)拷貝動態(tài)鏈接庫到系統(tǒng)共享目錄下,或在系統(tǒng)共享目錄

42、下為該動態(tài)鏈接庫建立個連接(硬連接或符號連接均可,常用符號連接).這里說的系統(tǒng)共享目錄,指的是LINUX動態(tài)鏈接庫存放的目錄,它包含/lib,/usr/lib以及/etc/ld.so.conf文件內所列的一系列目錄. # cp libmy.so /lib # ldconfig # 或: # ln -s pwd/libmy.so /lib # ldconfig # (2)將動態(tài)鏈接庫所在目錄名追加到動態(tài)鏈接庫配置文件/etc/ld.so.conf中. # pwd >> /etc/ld.so.conf # ldconfig # (3)利用動態(tài)鏈接庫管理命令ldconfig,強制其搜索指

43、定目錄,并更新緩存文件,便于動態(tài)裝入. # ldconfig pwd # 需要說明的是,這種操作方法雖然有效,但效果是暫時的,供程序測試還可以,一旦再度運行l(wèi)dconfig,則緩存文件內容可能改變,所需的動態(tài)鏈接庫可能不被系統(tǒng)共享了.與之相比較,前兩種方法是可靠的方法,值得業(yè)已定型的動態(tài)鏈接庫共享時采用.前兩種方法還有一個特點,即最后一條命令都是ldconfig,也即均需要更新一下緩存文件,以確保動態(tài)鏈接庫的共享生效. 四、含有動態(tài)函數的程序的編譯 4.1 防止編譯因未指定動態(tài)鏈接庫而出錯 當一個程序使用動態(tài)函數時,編譯該程序時就必須指定含所用動態(tài)函數的動態(tài)鏈接庫,否則編譯將會出錯退出.如本文

44、示例程序ady.c的編譯(未明確引用動態(tài)鏈接庫libmy.so): # cc -o ady ady.c /tmp/ccL4FsJp.o: In function main': /tmp/ccL4FsJp.o(.text+0x43): undefined reference to gettime' collect2: ld returned 1 exit status # 注: 因為ady.c所含的動態(tài)函數getdate,gettime不在系統(tǒng)函數庫中,所以連接時出錯. 4.2 編譯時引用動態(tài)鏈接庫的幾種方式 (1)當所用的動態(tài)鏈接庫在系統(tǒng)目錄(/lib,/usr/lib)下時,

45、可用編譯選項-l來引用.即: # cc -lmy -o ady ady.c # 注:編譯時用-l選項引用動態(tài)鏈接庫時,庫名須使用其縮寫形式.本例的my,表示引用libmy.so庫.若引用光標庫libncurses.so,須用-lncurses.注意,-l選項與參數之間不能有空格,否則會出錯. (2)當所用的動態(tài)鏈接庫在系統(tǒng)目錄(/lib,/usr/lib)以外的目錄時,須用編譯選項-L來指定動態(tài)鏈接庫所在的目錄(供編譯器查找用),同時用-l選項指定縮寫的動態(tài)鏈接庫名.即: # cc -L/usr/zzz/lib -lmy -o ady ady.c # (3)直接引用所需的動態(tài)鏈接庫.即: #

46、cc -o ady ady.c libmy.so # 或 # cc -o ady ady.c /lib/libmy.so # 等等.其中,動態(tài)鏈接庫的庫名可以采用相對路徑形式(文件名不以/開頭),也可采用絕對路徑形式(文件名以/開頭). 五、動態(tài)鏈接程序的運行與檢查 5.1 運行 編譯連接好含動態(tài)函數的程序后,就可以運行它了.動態(tài)鏈接程序因為共享了系統(tǒng)中的動態(tài)鏈接庫,所以其空間占用很小.但這并不意味功能的減少,它的執(zhí)行與靜態(tài)連接的程序執(zhí)行,效果完全相同.在命令提示符下鍵入程序名及相關參數后回車即可,如下例: $ ady 動態(tài)鏈接庫高級應用示范 當前日期: 2002-03-11 當前時間: 19:39:06 $ 5.2 檢查 檢查什么?檢查動態(tài)鏈接程序究竟需要哪些共享庫,系統(tǒng)中是否已有這些庫,沒有的話,用戶好想辦法把這些庫裝上. 怎么檢查呢?這里,告訴你一個實用程序-ldd,這個程序就是專門用來檢查動態(tài)鏈接程序依賴哪些共享庫的. ldd命令行用法如下: ldd -version -v|-verbose -d|-data-relocs -r|-function-relocs -help FILE.

溫馨提示

  • 1. 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請下載最新的WinRAR軟件解壓。
  • 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請聯(lián)系上傳者。文件的所有權益歸上傳用戶所有。
  • 3. 本站RAR壓縮包中若帶圖紙,網頁內容里面會有圖紙預覽,若沒有圖紙預覽就沒有圖紙。
  • 4. 未經權益所有人同意不得將文件中的內容挪作商業(yè)或盈利用途。
  • 5. 人人文庫網僅提供信息存儲空間,僅對用戶上傳內容的表現(xiàn)方式做保護處理,對用戶上傳分享的文檔內容本身不做任何修改或編輯,并不能對任何下載內容負責。
  • 6. 下載文件中如有侵權或不適當內容,請與我們聯(lián)系,我們立即糾正。
  • 7. 本站不保證下載資源的準確性、安全性和完整性, 同時也不承擔用戶因使用這些下載資源對自己和他人造成任何形式的傷害或損失。

評論

0/150

提交評論