博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Linux下动态加载SO文件
阅读量:7093 次
发布时间:2019-06-28

本文共 3107 字,大约阅读时间需要 10 分钟。

  hot3.png

动态库的显式调用

显式调用的含义是代码出现库文件名,用户需要自己去打开和管理库文件。其要点为:

  • dlfcn.h系统头文件包含进来
  • 用dlopen函数打开库文件,并指定打开方式, dllope的的第一个参数为共享库的名称,将会在下面位置查找指定的共享库。

                   ①环境变量LD_LIBRARY_PATH列出的用分号间隔的所有目录。

                   ②文件/etc/ld.so.cache中找到的库的列表,由ldconfig命令刷新。
                   ③目录usr/lib。
                   ④目录/lib。
                   ⑤当前目录。

                    第二个参数为打开共享库的方式。有两个取值

                        ①RTLD_NOW:将共享库中的所有函数加载到内存
                        ②RTLD_LAZY:会推后共享库中的函数的加载操作,直到调用dlsym()时方加载某函数

  • 用dlerror()函数测试是否打开成功,并进行错误处理;
  • 用dlsym获得函数地址,存放在一个函数指针中
  • 用获得的函数指针进行函数调用。
  • 程序结束时用dlclose关闭打开的动态库,防止资源泄露。
  • 用ldconfig工具把动态库的路径加到系统库列表中

1、编写测试文件

//main.c 测试动态库显式调用的程序#include
     //用于动态库管理的系统头文件  #include"myalib.h"    //要把函数的头文件包含进来,否则编译时会报错int main(intargc,char* argv[]){   //声明对应的函数的函数指针   void(*pTest)();           //加载动态库   void*pdlHandle = dlopen("libtest.so", RTLD_LAZY);   //错误处理       if(pdlHandle == NULL )    {       printf("Failed load 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 -o main   -ldl   main.c

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


3、执行测试程序

执行 ./main

输出   test

说明成功。


//say.c

 

#include 
int say(char **str){        printf("%s\n",str);}

将say.c 生成共享库的编译:

gcc   -o   dlopen.so   -shared   say.c

//使用dlopen函数动态加载库的源代码

#include 
#include
#include
void show_help(char *msg){        if(msg == NULL){                printf("Usage:mydlopen dlopen.so say stringtosay\n");        }else{                printf("%s\n",msg);        }        exit(1);}int main(int ac,char ** av){        if(ac < 3){                show_help(NULL);        }        void *handle;        //void *pfunc;        int (*pfunc)(char *str);        char * filename = av[1];        char * func = av[2];        char * word = av[3];        char * error;        handle = dlopen(filename,RTLD_LAZY);        if(!handle){                printf("Error: handle\n");                return 1;        }        pfunc = (int (*)(char *))dlsym(handle,func);        if((error=dlerror()) != NULL){                printf("Error: dlsym\n");                return 2;        }        (*pfunc)(word);        dlclose(handle);        return 0;}

//编译命令:

gcc  -o  mydlopen  mydlopen.c  -ldl

注意事项:

1.dlsym返回的指针是无类型的,要转换的指定的函数的类型。

2.使用函数指针时的写法: (*pfunc)(word); 不能直接写成pfunc(word);会段错误的
3.C还不支持默认参数,写show_help时不能给msg以NULL的默认值
3.编译时要使用共享库dl 其中有dlopen dlsynmdlerror dlclose 函数

使用命令:

./mydlopen ./dlopen.so say aaaabbbdddd


说明:pfunc = (int(*)(char *))dlsym(handle,func);

其中char *是将参数强转,因为int (*pfunc)(char*str);是这样定义的。

如果int (*pfunc)(char *str,int number);这样定义,那么强转的时候应该这样写:

pfunc = (int(*)(char *,int))dlsym(handle,func);


编译的时候的说明,比如现在的文件目录树如下:

1.work/

2.|--head.h

3.|--main.c

4.|--func.c

5.|--func2.c

这个时候欲将func2.c编译成动态链接库dlfunc.so,而func2.c中的函数用到了head.h和func.c里面的变量和函数,这个时候应该使用命令:

gcc -o  dlfunc.so   -shared    func2.c   func.c   -I ./

这样就生成了dlfunc.so。

然后编译连接主程序。

gcc -o test main.c  -ldl   -I./

就行了。

转载于:https://my.oschina.net/mskk/blog/686359

你可能感兴趣的文章
Python基础学习笔记(四:条件判断与缩进)
查看>>
OID,主键生成策略,PO VO DTO,get和load区别,脏检查,快照,java对象的三种状态
查看>>
Objective-C语言分类与协议
查看>>
洛谷P1983 车站分级
查看>>
mongodb
查看>>
统计难题
查看>>
Discrete Function(简单数学题)
查看>>
如何判断某版本的.NET Framework是否安装
查看>>
搭建app自动化测试环境(一)
查看>>
IE加载项
查看>>
区间最值查询问题
查看>>
Sencha Toucha 2.1 文件上传
查看>>
一些实用的方法,持续更新...
查看>>
[redis] 数据特性简单实验
查看>>
iOS开发-消息通知机制(NSNotification和NSNotificationCenter)
查看>>
vue打包后出现的.map文件
查看>>
前端应用框架(三)
查看>>
多线程的死锁
查看>>
定时任务框架Quartz-(一)Quartz入门与Demo搭建
查看>>
css导航栏
查看>>