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时,才会真正被系统卸载。
编写测试文件
#include
#include //用于动态库管理的系统头文件
#include"test.h" //要把函数的头文件包含进来,否则编译时会报错
intmain(int argc,char* argv[])
{
//声明对应的函数的函数指针
void(*pTest)();
//加载动态库
void*pdlHandle = dlopen("libtest.so", RTLD_LAZY);
//错误处理
if(pdlHandle== NULL ) {
printf("Failedload library\n");
return-1;
}
char*pszErr = dlerror();
if(pszErr!= NULL)
{
printf("%s\n",pszErr);
return-1;
}
//获取函数的地址
pTest= dlsym(pdlHandle, "test");
pszErr= dlerror();
if(pszErr!= NULL)
{
printf("%s\n",pszErr);
dlclose(pdlHandle);
return-1;
}
//实现函数调用
(*pTest)();
//程序结束时关闭动态库
dlclose(pdlHandle);
return0;
}
2、编译测试文件使用-ldl选项指明生成的对象模块需要使用共享库
gcc -omain -ldl main.c
执行完后就生成了一个main文件
3、执行测试程序
执行 ./main