solaris中有類似windows的DLL有關的函數嗎?
2024-07-26 00:29:38
供稿:網友
發信人: cpu (奔騰的心), 信區: Solaris
標 題: Re: solaris中有類似windows的DLL有關的函數
發信站: 華南網木棉站 (Fri Aug 7 19:36:46 1998), 轉信
【 在 irene (yoyo) 的大作中提到: 】
: 如題
呵呵當然有,window那套動東東很多機制都是抄unix的
比如system(32)/*.dll是就是仿照unix的/usr/lib/*.so
unix的/usr/lib/*.so就是unix的動態庫(dynamic library)
給程序動態鏈接用的,反之/usr/lib/*.a是靜態庫,程序編譯
鏈接時就將相關函數鏈入目標文件。
實際上泥cc -o yyy yyy.o -lXXX那個XXX就是告訴cc找/usr/lib/libXXX.so..
yyy.o聲明調用了libXXX.so中的函數,鏈接成功的目標文件yyy在運行時將動態
調用libXXX.so的函數,至于cc -o yyy yyy.o libxxx.a那就不同了呵呵
cc對-l參數的缺省鏈接方式是動態鏈接,即只鏈接符號,不鏈入函數實體。
對鏈接方式可man ld。
言歸正傳,既然有動態庫,就肯定有與之相關的函數,window有LoadLibrary,
偶solaris有dlopen,就是dynamic library open,window能讓泥做
土版DLL,偶solaris早就能讓泥自己生產.so了呵呵,下面具體說明怎樣調用
動態庫libXXX里的函數而又不需要在cc中指定-lXXX。
首先是dlopen,格式:
#include
void * dlopen(const char *pathname, int mode);
返回一個void *類型的handle,否則返回NULL。
pathname就是泥所要打開的動態庫,如果這個庫聲明鏈接了其它庫,即對其它
庫有依賴關系,那么所有相關有依賴關系的庫都會被打開,這些打開的庫稱之
為組(group)。
mode是打開方式:
RTLD_LAZY:打開動態庫后只重定位庫中數據地址引用而不重定位而函數引用,
函數引用在該函數要被激活時才定位,的確LAZY呵呵,但省開銷;)
RTLD_NOW: 與上者相比,動態庫一被打開就重定位所有函數的引用。
RTLD_GLOBAL:打開動態庫里的全局符號可以被其它所有庫重定位。
RTLD_LOCAL: 打開動態庫里的全局符號只能被同組庫重引用。
RTLD_GROUP: 只有相關組的符號才允許重定位??
RTLD_PARENT:發dlopen調用的對象中的符號對被dlopen對象可見。
RTLD_WORLD: 。。。呵呵太晦澀了我翻譯的我都看不明白;(
總之,一個RTLD_LAZY已經夠用了呵呵;)
然后是得到重定位的數據或函數引用:
#include
void *dlsym(void *handle, const char *name)
意義明顯,handle即dlopen的返回值,name即泥要引用的在動態庫變量或函
數名稱。成功返回重定位后的符號地址,失敗返回NULL。
最后是關閉動態庫:int dlclose(void *handle),
一看就明白,懶得解釋了;)
下面給一個例子增加感性認識,該例子調用動態庫client.so中的函數
int client_request(char *),該函數返回0或-1并根據不同錯誤設置
字符串err_info(也定義在client.so中):
# include
# include
# include
# define TRUE 0
# define FALSE -1
main( )
{
char buf[64];
void *handle; /* 動態庫句柄 */
char *err_info; /* 要引用的動態庫中的一個變量 */
int (*client_request)(char *); /* 要引用的一個函數 */
/* 打開動態庫client.so */
if ((handle = dlopen("client.so", RTLD_LAZY)) == NULL) {
perror("dlopen");
exit(-1);
}
/* 得到函數名client_request的引用 */
if ((client_request =
(int (*)(char *))dlsym(handle, "client_request")) == NULL) {
perror("dlsym client_request");
exit(-1);
}
/* 得到變量名err_info的引用 */
if ((err_info =
(char *)dlsym(handle, "err_info")) == NULL) {
perror("dlsym err_info");
exit(-1);
}
for(;;) {
gets(buf); /* 從標準輸入讀入命令串 */
if (strcmp(buf, "exit") == TRUE) {
dlclose(handle); /* 關閉動態庫 */
return 0;
}
PRintf("request:%s/n", buf);
client_request(buf); /* 調用動態庫中的函數 */
printf("ask: %s/n", err_info); /* 引用動態庫中的變量 */
}
}
最后是編譯問題,怎樣編譯成.so文件呢?很簡單用ld或者cc -G就可以了,比如:
cc -G yyy.so yyy.o others.o -ldl
如果這個yyy.o有引用了其它動態庫的函數呢?那么用cc -G ... -l了
比如yyy.o引用了socket函數,那么
cc -G yyy.so yyy.o others.o -ldl -lsocket即可。