10.動態庫的顯式調用
庫函數dlopen()將打開一個新庫,並把它裝入內存。該函數主要用來加載庫中的符號,這些符號在編譯的時候是不知道的。比如Apache Web服務器利用這個函數在運行過程中加載模塊,這為它提供了額外的能力。一個配置文件控制了加載模塊的過程。這種機制使得在系統中添加或者刪除一個模塊時,都不需要重新編譯了。
dlopen()在dlfcn.h中定義,並在dl庫中實現。它需要兩個參數:一個文件名和一個標志。文件名可以是我們學習過的庫名字。標志指明是否立刻計算庫的依賴性。如果設置為RTLD_NOW的話,則立刻計算;如果設置的是RTLD_LAZY,則在需要的時候才計算。另外,可以指定RTLD_GLOBAL,它使得那些在以後才加載的庫可以獲得其中的符號。
當庫被裝入後,可以把 dlopen()返回的句柄作為給 dlsym()的第一個參數,以獲得符號在庫中的地址。使用這個地址,就可以獲得庫中特定函數的指針,並且調用裝載庫中的相應函數。
下面詳細說明一下這些函數。
dlerror
原型為:const char *dlerror(void);
當動態鏈接庫操作函數執行失敗時,dlerror可以返回出錯信息,返回值為NULL時表示操作函數執行成功。
dlopen
原型為:void *dlopen (const char *filename, int flag);
dlopen用於打開指定名字(filename)的動態鏈接庫,並返回操作句柄。
filename:如果名字不以/開頭,則非絕對路徑名,將按下列先後順序查找該文件。
(1)用戶環境變量中的LD_LIBRARY值;
(2)動態鏈接緩沖文件/etc/ld.so.cache
(3)目錄/lib,/usr/lib
flag表示在什麼時候解決未定義的符號(調用)。取值有兩個:
1)RTLD_LAZY :表明在動態鏈接庫的函數代碼執行時解決。
2)RTLD_NOW :表明在dlopen返回前就解決所有未定義的符號,一旦未解決,dlopen將返回錯誤。
dlopen調用失敗時,將返回NULL值,否則返回的是操作句柄。
dlsym
取函數執行地址
原型為:void *dlsym(void *handle, char *symbol);
dlsym根據動態鏈接庫操作句柄(handle)與符號(symbol),返回符號對應的函數的執行代碼地址。由此地址,可以帶參數執行相應的函數。
如程序代碼:void (*add)(int x,int y); /*說明一下要調用的動態函數add*/
add=dlsym("xxx.so","add");/*打開xxx.so共享庫,取add函數地址*/
add(89,369);/*帶兩個參數89和369調用add函數*/
dlclose:關閉動態鏈接庫
原型為:int dlclose (void *handle);
dlclose用於關閉指定句柄的動態鏈接庫,只有當此動態鏈接庫的使用計數為0時,才會真正被系統卸載。