前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >应用程序设计:在动态库中如何调用外部函数?

应用程序设计:在动态库中如何调用外部函数?

作者头像
IOT物联网小镇
发布于 2021-06-22 10:33:15
发布于 2021-06-22 10:33:15
3K00
代码可运行
举报
文章被收录于专栏:IOT物联网小镇IOT物联网小镇
运行总次数:0
代码可运行

大家好,我是一个动态链接库!

这个名字,相信你一定早就如雷贯耳了。

在计算机早期时代,由于内存资源紧张,我可是发挥了重大的作用!

不论是在 Windows 系统中,还是在 Unix 系列平台上,到处都能见到我的身影,因为我能为大家节省很多资源啊,资源就是人民币!

愉快的玩耍

比如:我的主人编写了这么一段简单的代码:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
# 文件:lib.c

#include <stdio.h>

int func_in_lib(int k)
{
    printf("func_in_lib is called \n");
    return k + 1;
}

只要用如下命令来编译,我就诞生出来了 lib.so,也就是一个动态链接库:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
$ gcc -m32 -fPIC --shared -o lib.so lib.c

这个时候,主人随便把我丢给谁,我都可以为他服务,只要他调用我肚子里的这个函数 func_in_lib 就可以了。

虽然目前你看到我提供的这个函数很简单,但是道理都是一样的,后面如果有机会,我就在这个函数里来计算机器人的运动轨迹,给你瞧一瞧!

例如:张三今天写了一段代码,需要调用我的这个函数。

张三这个人比较喜欢骚操作,明明他在编译可执行程序的时候,把我动态链接一下就可以了,就像下面这样:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
$ gcc -m32 -o main main.c ./lib.so

但是张三偏偏不这么做,为了炫技,他选择使用 dlopen 动态加载的方式,来把我从硬盘上加载到进程中。

咱们来一起围观一下张三写的可执行程序代码:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
# 文件:main.c

#include <unistd.h>
#include <stdio.h>
#include <stdlib.h>
#include <dlfcn.h>

typedef int (*pfunc)(int);

int main(int argc, char *agv[])
{
    int a = 1;
    int b;

    // 打开动态库
    void *handle = dlopen("./lib.so", RTLD_NOW);
    if (handle)
    {
        // 查找动态库中的函数
        pfunc func = (pfunc) dlsym(handle, "func_in_lib");
        if (func)
        {
            b = func(a);
            printf("b = %d \n", b);
        }
        else
        {
            printf("dlsym failed! \n");
        }
        dlclose(handle);
    }
    else
    {
        printf("dlopen failed! \n");
    }
    
    return 0;
}

从代码中可以看到,张三预先知道我肚子里的这个函数名称是 func_in_lib,所以他使用了系统函数 dlsym(handle, "func_in_lib"); 来找到这个函数在内存中的加载地址,然后就可以直接调用这个函数了。

张三编译得到可执行文件 main 之后,执行结果完全正确,很开心!

悲从中来

可是有一天,我遇到一件烦人的事情,我的主人说:你这个服务函数的计算过程太单调了,给你找点乐子,你在执行的时候啊,到其他一个外部模块里调用一个函数。

话刚说完,就丢给我一个函数名:void func_in_main(void);

也就是说,我需要在我的服务函数中,去调用其他模块里的函数,就像下面这样:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
#include <stdio.h>

// 外部函数声明
void func_in_main(void);

int func_in_lib(int k)
{
    printf("func_in_lib is called \n");

    // 调用外部函数
    func_in_main();
    
    return k + 1;
}

那么这个函数在哪里呢?天哪,我怎么知道这个函数是什么鬼?怎么才能找到它藏在内存的那个角落(地址)里?

不管怎么样,主人修改了代码之后,还是很顺利的把我编译了出来:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
$ gcc -m32 -fPIC --shared -o lib.so lib.c

编译指令完全没有变化。

因为我仅仅是一个动态链接库,这个时候即使我不知道 func_in_main 函数的地址,也是可以编译成功的。

只不过我要把这个家伙标记一下:谁要是想使用我,就必须告诉我这个家伙的地址在哪里!,否则就别怪我耍赖。

无辜的张三

我的主人对张三说:兄弟,我的这个动态链接库升级了,功能更强大哦,想不想试一下?

张三心想:我是使用 dlopen 的方式来动态加载动态库文件的,不需要对可执行程序重新编译或者链接,直接运行就完事了!

于是他二话不说,直接就把我拿过去,丢在他的可执行程序目录下,然后执行 main 程序。

可是这一次,他看到的结果却是:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
dlopen failed!

为什么会加载失败呢?上次明明是正常执行的!张三一脸懵逼!

其实,这压根就不能怪我!以为我刚才就说了:谁要是想使用我,就必须告诉我 func_in_main 这个函数的地址在哪里!

可是在张三的这个进程里,我到处都找不到这个函数的地址。既然你没法满足我,那我就没法满足你!

锦囊1: 导出符号表

张三这下也没辙了,只要找我的主人算账:我的应用程序代码一丝一毫都没有动,怎么换了你给的新动态链接库就不行了呢?

主人慢条斯理的回答:疏忽了,疏忽了,忘记跟你说一件事情了:这个动态库啊,它需要你多做一件事情:在你的程序中提供一个名为 func_in_main 的函数,这样就可以了。

张三一想:这个好办,加一个函数就是了。

因为这个可执行程序只有一个 main.c 文件,于是他在其中新加了一个函数:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
void func_in_main(void)
{
    printf("func_in_main \n");
}

然后就开始编译、执行,一顿操作猛如虎:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
# gcc -m32 -o main main.c -ldl
# ./main
dlopen failed!

咦?怎么还是失败?!已经按照要求加了 func_in_main 这个函数了啊?!

这个傻X张三,对,你确实是在 main.c 中加了这个函数,但是你仅仅是加在你的可执行程序中的,但是我却压根就看不到这个函数啊!

不信的话,你检查一下编译出来的可执行程序中,是否把 func_in_main 这个符号导出来了?如果不导出来,我怎么能看到?

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
# 查看导出的符号表
$ objdump -e main -T | grep func_in_main
# 这里输出为空

既然输出为空,就说明没有导出来!这个就不用我教你了吧?

茴香豆的“茴”字,一共有四种写法。。。

哦,不,导出符号,一共有两种方式:

方式1:导出所有的符号

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
$ gcc -m32 -rdynamic -o main main.c -ldl

当然,下面这个指令也可以:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
gcc -m32 -Wl,--export-dynamic -o main main.c -ldl

方式2:导出指定的符号

先定义一个文件,把需要导出的符号全部罗列出来:

文件:exported.txt

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
{
    extern "C"
    {
        func_in_main;
    };
};

然后,在编译选项中指定这个导出文件:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
gcc -m32 -Wl,-dynamic-list=./exported.txt -o main main.c -ldl

使用以上两种方式的任意一种即可,编译之后,再使用 objdump 指令看一下导出符号:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
$ objdump -e main -T | grep func_in_main
080485bb g    DF .text	00000019  Base        func_in_main

嗯,很好很好!张三赶紧按照这样的方式操作了一下,果真成功执行了函数!

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
$ ./main 
func_in_lib is called 
func_in_main 
b = 2

也就是说,在我的动态库文件中,正确的找到了外部其他模块中的函数地址,并且愉快的执行成功了!

锦囊2: 动态注册

虽然执行成功了,张三的心里隐隐约约的仍然有一丝不爽的感觉,每次编译都要导出符号,真麻烦,能不能优化一下?

于是他找到我的主人,表达了自己的不满。

主人一瞧,有个性!既然你不想提供,那我就满足你:

  1. 首先,在动态库中提供一个默认的函数实现(func_in_main_def);
  2. 然后,再提供一个专门的注册函数(register_func),如果外部模块想提供 func_in_main 这个函数,就调用注册函数注册进来;

此时,lib.c 最新的代码就变成这个样子了:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
#include <stdio.h>

// 默认实现
void func_in_main_def(void)
{
    printf("the main is lazy, do NOT register me! \n");
}

// 定义外部函数指针
void (*func_in_main)() = func_in_main_def;

void register_func(void (*pf)())
{
    func_in_main = pf;
}

int func_in_lib(int k)
{
    printf("func_in_lib is called \n");

    if (func_in_main)
        func_in_main();

    return k + 1;
}

然后编译,全新的我再一次诞生了 lib.so

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
gcc -m32 -fPIC --shared -o lib.so lib.c

主人把我丢给张三的时候说:好了,满足你的需求,这一次你不用提供 func_in_main 这个函数了,当然也就不用再导出符号了。

不过,如果如果有一天,你改变了注意,又想提供这个函数了,那么你就要通过动态库中的 register_func 函数,把你的函数注册进来。

Have you got it?赶紧再去试一下!

这个时候,张三再次使用我的时候,就不需要导出他的 main.c 里的那个函数 func_in_main 了,实际上他可以把这个函数从代码中删掉!

编译、执行,张三再一次猛如虎的操作:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
$ gcc -m32 -o main main.c -ldl
$ ./main
func_in_lib is called 
the main is lazy, do NOT register me! 
b = 2

嗯,结果看起来是正确的。

咦?怎么多了一行字:the main is lazy, do NOT register me!

难道是在质疑我的技术能力吗?好吧,既然如此,我也满足你,不就是注册一个函数嘛,简单:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
// 文件: main.c

#include <unistd.h>
#include <stdio.h>
#include <stdlib.h>
#include <dlfcn.h>

typedef int (*pfunc)(int);
typedef int (*pregister)(void (*)());

// 控制注册函数的宏定义
#define REG_FUNC

#ifdef REG_FUNC
void func_in_main(void)
{
    printf("func_in_main \n");
}
#endif

int main(int argc, char *agv[])
{
    int a = 1;
    int b;

    // 打开动态库
    void *handle = dlopen("./lib.so", RTLD_NOW);
    if (handle)
    {
#ifdef REG_FUNC
        // 查找动态库中的注册函数
        pregister register_func = (pregister) dlsym(handle, "register_func");
        if (register_func)
        {

            register_func(func_in_main);
        }
#endif

        // 查找动态库中的函数
        pfunc func = (pfunc) dlsym(handle, "func_in_lib");
        if (func)
        {
            b = func(a);
            printf("b = %d \n", b);
        }
        else
        {
            printf("dlsym failed! \n");
        }
        dlclose(handle);
    }
    else
    {
        printf("dlopen failed! \n");
    }
    
    return 0;
}

然后编译、执行:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
$ gcc -m32 -o main main.c -ldl
$ ./main 
func_in_lib is called 
func_in_main 
b = 2

完美收官!

PS:很多平台级的代码,例如一些工控领域的运行时(Runtime)软件,大部分都是通过注册的方式,来把平台代码、用户代码进行连接、绑定的。 ------ End ------

本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2021-05-21,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 IOT物联网小镇 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
高性能服务器架构思路(四)——编码复杂度和通信
以前我们的代码,从上往下执行,每一行都会占用一定的 CPU 时间,这些代码的执行顺序,也是和编写的顺序基本一致,任何一行代码,都是唯一时刻的执行任务。当我们在编写分布式程序的时候,我们的代码将不再好像那些单进程、单线程的程序一样简单。我们要把同时运行的不同代码,在同一段代码中编写。就好像我们要把整个交响乐团的每个乐器的乐谱,全部写到一张纸上。为了解决这种编程的复杂度,业界发展出了多种编码形式。
2019/06/30
4320
自己的电脑手机作为服务器,搭建WEB/FTP,并通过域名访问
要求:1.    一台服务器(电脑可以使用虚拟机,手机安卓可以使用KSWAB,苹果未知) 推荐使用闲置的安卓手机,废物利用,而且还不耗电。(KSWAB正版需要购买(推荐),但吾爱也有破解版) 2.    一台支持转发的家用路由器(我使用的是斐讯K2,自带的固件) 注:最好是常年不会关机的路由器,因为路由器重启会获取新的公网IP(获取动态IP非静态),解析的时候就需要重新配置,麻烦。 3.    能设置解析的域名(没有的自行百度注册域名) 注:此方法最好是不要使用已经备案的域名,可能会被工信部查出来,注销你已经备案号,得不偿失(理论上)
凯哥Java
2019/06/28
6.2K0
自己的电脑手机作为服务器,搭建WEB/FTP,并通过域名访问
高性能服务器架构思路(五)——分布式缓存
在分布式程序架构中,如果我们需要整个体系有更高的稳定性,能够对进程容灾或者动态扩容提供支持,那么最难解决的问题,就是每个进程中的内存状态。因为进程一旦毁灭,内存中的状态会消失,这就很难不影响提供的服务。所以我们需要一种方法,让进程的内存状态,不太影响整体服务,甚至最好能变成“无状态”的服务。当然“状态”如果不写入磁盘,始终还是需要某些进程来承载的。在现在流行的 WEB 开发模式中,很多人会使用 PHP+Memcached+MySQL 这种模型,在这里,PHP 就是无状态的,因为状态都是放在 Memcached 里面。这种做法对于 PHP 来说,是可以随时动态的毁灭或者新建,但是 Memcached 进程就要保证稳定才行;而且 Memcached 作为一个额外的进程,和它通信本身也会消耗更多的延迟时间。因此我们需要一种更灵活和通用的进程状态保存方案,我们把这种任务叫做“分布式缓存”的策略。我们希望进程在读取数据的时候,能有最高的性能,最好能和在堆内存中读写类似,又希望这些缓存数据,能被放在多个进程内,以分布式的形态提供高吞吐的服务,其中最关键的问题,就是缓存数据的同步。
2019/06/30
3940
腾讯云服务器完美搬家步骤教程
新客户无门槛领取总价值高达2860元代金券,每种代金券限量500张,先到先得。 http://cloud.tencent.com/act/pro/voucherslist 1.首先登入腾讯云服务器管理
用户5803396
2019/07/12
10.5K0
腾讯云怎么领取免费云服务器?
腾讯云是腾讯提供的网络平台,现在注册一个腾讯云可以领取一个云服务器。注册的步骤咗嚛就不介绍了,登录腾讯云后台个活动,点击领取个人免费套
主机优惠教程
2019/04/25
118.3K0
腾讯云怎么领取免费云服务器?
微擎安装之腾讯云懒人教程篇
自己去官网买要1715.64 点击下面的活动链接只要720 但是有一个限制那就是新用户。可以拿父母兄弟姐妹的账号来买。
小龙
2018/10/17
8K0
建站教程:云服务器使用(完整版)
开头大家可以领取下优惠券;最高可领取2860,反正免费,万一以后用得上呢,不要白不要新客户无门槛领取总价值高达2860元代金券,每种代金券限量500张,先到先得,点击链接可领取。
2019/06/30
5.1K0
高性能服务器架构思路(一)——缓冲策略
在服务器端程序开发领域,性能问题一直是备受关注的重点。业界有大量的框架、组件、类库都是以性能为卖点而广为人知。然而,服务器端程序在性能问题上应该有何种基本思路,这个却很少被这些项目的文档提及。本文正式希望介绍服务器端解决性能问题的基本策略和经典实践,并分为几个部分来说明:
2019/06/30
5340
高性能服务器架构思路(二)——缓冲清理策略
虽然使用缓存思想似乎是一个很简单的事情,但是缓存机制却有一个核心的难点,就是——缓存清理。我们所说的缓存,都是保存一些数据,但是这些数据往往是会变化的,我们要针对这些变化,清理掉保存的“脏”数据,却可能不是那么容易。
2019/06/30
8520
腾讯云服务器+宝塔面板+wordpress搭建网站教程
建网站本身是没有什么难度的,只是配套需要的东西有些繁琐,我们需要提前准备一些软件之类的必备的东西,其中主要包括:
云优惠
2019/09/12
8.2K0
腾讯云服务器+宝塔面板+wordpress搭建网站教程
怎么使用腾讯云服务器搭建个人网站?
【新用户限量秒杀】云服务器限时秒杀,首购1核1G 99元/年 https://cloud.tencent.com/act 从刚开始的简单学习HTML语言,到进入实验室跟着老师,学长学习Java,An
云优惠
2019/09/11
33.6K0
怎么使用腾讯云服务器搭建个人网站?
腾讯云服务器从购买到配置(Linux系统、新手教程,图文教程)
1.首先我们来到腾讯云的官网,若是学生的话,则可以在合作与生态里选择“云+校园”这个选项↓
JunyouYH
2019/07/17
3K0
【热】只有100元预算,如何在腾讯云上建设一个1年的Web网站??
大家好,相信很多的同学都是用过腾讯云,并且都有感情了,今天为大家分享一个经验,如何使用100元预算,建一个1年的Web网站!!!废话少说,直接上干货:
TCS-F
2021/10/21
6.4K0
【热】只有100元预算,如何在腾讯云上建设一个1年的Web网站??
轻松入门腾讯云存储:对象存储COS的基本功能详解
腾讯云是全球领先的云计算服务商之一,将腾讯集团在QQ、微信、QQ空间等业务中积累的海量互联网服务能力,开放给各行各业,并不断输出计算机视觉、智能语音、大数据分析、安全防护、音视频等业界领先的智能科技,为政务、金融、电商、零售、视频、出行、汽车、工业、能源等不同行业打造科技高速路,并携手合作伙伴一同创建智慧的数字生态。
勤劳的小蜜蜂
2019/06/29
3.6K0
怎么使用腾讯云服务器搭建个人网站?
【新用户限量秒杀】云服务器限时秒杀,首购1核1G 99元/年 https://cloud.tencent.com/act 从刚开始的简单学习HTML语言,到进入实验室跟着老师,学长学习Java,An
JunyouYH
2019/09/18
31.7K0
腾讯云服务器web环境配置教程
腾讯云服务器是现在越来越多站长以及企业建站的首选了,因为腾讯云背靠腾讯这颗大树,有10亿微信用户和10亿QQ用户流量考验验做为背书,有足够的技术实力和奖金实力让人相信它是靠谱的。双10亿的用户流量都能把持得住,还在乎你个小网站的流量压力吗?
勤劳的小蜜蜂
2019/06/17
12.7K0
腾讯云服务器web环境配置教程
腾讯云搭建WordPress网站教程详解
建网站本身是没有什么难度的,只是配套需要的东西有些繁琐,我们需要提前准备一些软件之类的必备的东西,其中主要包括:
主机精选@主机点评
2019/09/19
8.7K0
建站大促惊喜上线!还有你们久等的 Serverless 域名备案答疑!
我们都知道使用中国内地(大陆)的 Serverless 服务开办网站并绑定域名服务时必须先办理网站备案,在过去,备案前大多数都需要先购买一台服务器,而现在,你有了最低成本的备案选择:Serverless 备案!赶快前往 Serverless&DNSPod 联合推出的免费建站活动: https://cloud.tencent.com/act/pro/serverless-dnspod ,购买性价比最高的 Serverless 备案资源包和冰点折扣域名解析套餐吧! Serverless 建站免费资源大合集
腾讯云DNSPod团队
2021/01/22
7K0
腾讯云CDN接入教程
【数字生态,钜惠来袭】云服务器限时秒杀,首购1核1G 99元/年 http://cloud.tencent.com/act/pro/voucherslist 开通CDN服务 开通路径,顶部导航条—基础
勤劳的小蜜蜂
2019/06/27
10.3K0
腾讯云服务器优惠活动,腾讯云服务器最新优惠活动整理
最近,很多用户问我腾讯云现在有哪些优惠,怎么购买才最划算,今天特意做了下整理,希望能给大家带来帮助。
tengxunyun
2018/10/20
42.8K2
腾讯云服务器优惠活动,腾讯云服务器最新优惠活动整理
推荐阅读
相关推荐
高性能服务器架构思路(四)——编码复杂度和通信
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档