Linux 动态库的显示调用

来源:互联网 发布:中科院心理研究所知乎 编辑:程序博客网 时间:2024/06/15 02:11

Linux 动态库的显示调用

分类: 动态库与静态库 1710人阅读 评论(0)收藏 举报
linuxnulllibrary测试web服务apache

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);/*带两个参数89369调用add函数*/

  • dlclose:关闭动态链接库

原型为:int dlclose (void *handle);

dlclose用于关闭指定句柄的动态链接库,只有当此动态链接库的使用计数为0,才会真正被系统卸载。


编写测试文件

[html] view plaincopy
  1. #include<stdio.h>  
  2.   
  3. #include<dlfcn.h> //用于动态库管理的系统头文件  
  4.   
  5. #include"test.h" //要把函数的头文件包含进来,否则编译时会报错  
  6.   
  7. intmain(int argc,char* argv[])  
  8.   
  9. {  
  10.   
  11. //声明对应的函数的函数指针  
  12.   
  13. void(*pTest)();  
  14.   
  15. //加载动态库  
  16.   
  17. void*pdlHandle = dlopen("libtest.so", RTLD_LAZY);  
  18.   
  19. //错误处理  
  20.   
  21. if(pdlHandle== NULL ) {  
  22.   
  23. printf("Failedload library\n");  
  24.   
  25. return-1;  
  26.   
  27. }  
  28.   
  29. char*pszErr = dlerror();  
  30.   
  31. if(pszErr!= NULL)  
  32.   
  33. {  
  34.   
  35. printf("%s\n",pszErr);  
  36.   
  37. return-1;  
  38.   
  39. }  
  40.   
  41. //获取函数的地址  
  42.   
  43. pTestdlsym(pdlHandle, "test");  
  44.   
  45. pszErrdlerror();  
  46.   
  47. if(pszErr!= NULL)  
  48.   
  49. {  
  50.   
  51. printf("%s\n",pszErr);  
  52.   
  53. dlclose(pdlHandle);  
  54.   
  55. return-1;  
  56.   
  57. }  
  58.   
  59. //实现函数调用  
  60.   
  61. (*pTest)();  
  62.   
  63.   
  64. //程序结束时关闭动态库  
  65.   
  66. dlclose(pdlHandle);  
  67.   
  68. return0;  
  69.   
  70. }   

2、编译测试文件使用-ldl选项指明生成的对象模块需要使用共享库

gcc -omain -ldl main.c

执行完后就生成了一个main文件


3、执行测试程序

执行 ./main

0 0