显式运行时链接

dlopen
功能:打开一个动态链接库
包含头文件:
#include <dlfcn.h>
函数定义:
void * dlopen( const char * pathname, int mode);
函数描述:
在dlopen()函数以指定模式打开指定的动态连接库文件,并返回一个句柄给调用进程。使用dlclose()来卸载打开的库。
mode是打开方式,其值有多个,不同操作系统上实现的功能有所不同,在linux下,按功能可分为三类:
1、解析方式
RTLD_LAZY:在dlopen返回前,对于动态库中的未定义的符号不执行解析(只对函数引用有效,对于变量引用总是立即解析)。
RTLD_NOW: 需要在dlopen返回前,解析出所有未定义符号,如果解析不出来,在dlopen会返回NULL,错误为:: undefined symbol: xxxx.......
2、作用范围,可与解析方式通过“|”组合使用。
RTLD_GLOBAL:动态库中定义的符号可被其后打开的其它库重定位。
RTLD_LOCAL: 与RTLD_GLOBAL作用相反,动态库中定义的符号不能被其后打开的其它库重定位。如果没有指明是RTLD_GLOBAL还是RTLD_LOCAL,则缺省为RTLD_LOCAL。
3、作用方式
RTLD_NODELETE: 在dlclose()期间不卸载库,并且在以后使用dlopen()重新加载库时不初始化库中的静态变量。这个flag不是POSIX-2001标准。
RTLD_NOLOAD: 不加载库。可用于测试库是否已加载(dlopen()返回NULL说明未加载,否则说明已加载),也可用于改变已加载库的flag,如:先前加载库的flag为RTLD_LOCAL,用dlopen(RTLD_NOLOAD|RTLD_GLOBAL)后flag将变成RTLD_GLOBAL。这个flag不是POSIX-2001标准。
RTLD_DEEPBIND:在搜索全局符号前先搜索库内的符号,避免同名符号的冲突。这个flag不是POSIX-2001标准。
返回值:
打开错误返回NULL
成功,返回库引用
编译时候要加入 -ldl (指定dl库)

dlsym
功能:
根据动态链接库操作句柄与符号,返回符号对应的地址。
包含头文件:
#include <dlfcn.h>
函数定义:
void*dlsym(void* handle,const char* symbol)
函数描述:
dlsym根据动态链接库操作句柄(handle)与符号(symbol),返回符号对应的地址。使用这个函数不但可以获取函数地址,也可以获取变量地址。
handle是由dlopen打开动态链接库后返回的指针,symbol就是要求获取的函数或全局变量的名称。
返回值:
1、如果查找的符号是个函数,那么它返回函数的地址
2、如果查找的符号是个变量,那么它返回变量的地址
3、如果查找的符号是个常量,那么它返回的是该常量的值

dlerror
每次我们调用dlopen()、dlsym()或dlclose()以后,我们都可以调用dlerror()函数老判断上一次调用是否成功。dlerror的返回类型是char *,如果返回NULL,则表示上一次调用成功;如果不是,则返回相应的错误信息。

dlclose
dlclose的作用跟dlopen刚好相反,它的作用是将一个已经加载的模块卸载。系统会维持一个加载引用计数器,每次使用dlopen加载某模块时,相应的计数器加1;每次使用dlclose卸载某模块时,相应计数器减1.只有当计数器值减到0时,模块才被真正地卸载掉。卸载的过程跟加载刚好相反,先执行“.finit" 段的代码,然后将相应的符号从符号表中去除,取消进程空间跟模块的映射关系,然后关闭模块文件。

程序示例如下(RunSoSimple.c):
#include<stdio.h>
#include<dlfcn.h>
int main(int argc, char * argv[])
{
      void * handle;
      double (*func) (double);
      char * error;
      handle = dlopen(argv[1], RTLD_NOW);
      if(handle == NULL)
      {
            printf("Open library %s error: %s\n",argv[1], dlerror() );
            return -1;
      }
      func = dlsym(handle, "sin");
      if( (error = dlerror() ) != NULL )
      {
            printf("Symbol sin not found: %s\n",error);
            goto exit_runso;
      }
      printf("%f\n",func(3.1415926 / 2) );
exit_runso:
      dlclose(handle);
      return 0;
}
编译运行程序
gcc -g RunSoSimple.c -o RunSoSimple -ldl
./RunSoSimple /lib/i386-linux-gnu/libm-2.13.so
结果如下:
1.000000
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章