iis服务器助手广告广告
返回顶部
首页 > 资讯 > 前端开发 > node.js >在动态库中调用外部函数的方法
  • 883
分享到

在动态库中调用外部函数的方法

2024-04-02 19:04:59 883人浏览 薄情痞子
摘要

本篇内容介绍了“在动态库中调用外部函数的方法”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!愉快的玩耍比如:

本篇内容介绍了“在动态库中调用外部函数的方法”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!

愉快的玩耍

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

# 文件:lib.c  #include <stdio.h>  int func_in_lib(int k) {     printf("func_in_lib is called \n");     return k + 1; }

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

GCc -m32 -fPIC --shared -o lib.so lib.c

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

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

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

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

$ gcc -m32 -o main main.c ./lib.so

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

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

# 文件: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);。

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

#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; }

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

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

$ gcc -m32 -fPIC --shared -o lib.so lib.c

编译指令完全没有变化。

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

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

无辜的张三

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

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

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

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

dlopen failed!

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

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

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

锦囊1: 导出符号表

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

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

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

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

void func_in_main(void) {     printf("func_in_main \n"); }

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

# gcc -m32 -o main main.c -ldl # ./main dlopen failed!

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

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

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

# 查看导出的符号表 $ objdump -e main -T | grep func_in_main # 这里输出为空

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

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

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

方式1:导出所有的符号

$ gcc -m32 -rdynamic -o main main.c -ldl

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

gcc -m32 -Wl,--export-dynamic -o main main.c -ldl

方式2:导出指定的符号

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

文件:exported.txt

{     extern "C"     {         func_in_main;     }; };

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

gcc -m32 -Wl,-dynamic-list=./exported.txt -o main main.c -ldl

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

$ objdump -e main -T | grep func_in_main 080485bb g    DF .text  00000019  Base        func_in_main

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

$ ./main  func_in_lib is called  func_in_main  b = 2

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

锦囊2: 动态注册

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

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

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

  1. 鸿蒙官方战略合作共建——HarmonyOS技术社区

  2. 首先,在动态库中提供一个默认的函数实现(func_in_main_def);

  3. 然后,再提供一个专门的注册函数(reGISter_func),如果外部模块想提供 func_in_main 这个函数,就调用注册函数注册进来;

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

#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:

gcc -m32 -fPIC --shared -o lib.so lib.c

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

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

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

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

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

$ 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!

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

// 文件: 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; }

然后编译、执行:

$ gcc -m32 -o main main.c -ldl $ ./main  func_in_lib is called  func_in_main  b = 2

完美收官!

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

“在动态库中调用外部函数的方法”的内容就介绍到这里了,感谢大家的阅读。如果想了解更多行业相关的知识可以关注编程网网站,小编将为大家输出更多高质量的实用文章!

--结束END--

本文标题: 在动态库中调用外部函数的方法

本文链接: https://www.lsjlt.com/news/66851.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • 在动态库中调用外部函数的方法
    本篇内容介绍了“在动态库中调用外部函数的方法”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!愉快的玩耍比如:...
    99+
    2022-10-19
  • C#调用C++动态库接口函数和回调函数方法
    目录1. 前言2. 普通接口函数调用示例2.1 C++端编写接口2.2 C#端调用3. 回调函数调用示例3.1 C++端编写接口3.2 C#端调用1. 前言 需求: 当前C已经写好了...
    99+
    2022-11-13
  • smarty使用外部函数的方法
    这篇文章主要介绍smarty使用外部函数的方法,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!  1.系统函数使用:<!DOCTYPE html><html><head...
    99+
    2023-06-07
  • php中为什么无法在外部静态地调用类和方法
    PHP是一种广泛使用的编程语言,以其简洁易用的语法、快速的开发速度和支持多种数据库为特色。但是,有一些PHP开发者可能会遇到一个问题:无法在外部静态地调用类和方法。本文将探讨这个问题,并提供解决方案。什么是静态调用?在PHP开发中,静态调用...
    99+
    2023-05-14
    php
  • Go调用Rust方法及外部函数接口前置
    目录前言FFI 和 Binding准备 Rust 示例程序用 Cargo 创建项目准备 Rust 代码编译 Rust 代码准备 Go 代码编写 main.go编译代码总结前言 近期 ...
    99+
    2022-11-13
  • php无法在外部静态调用类和方法的原因是什么
    这篇文章主要讲解了“php无法在外部静态调用类和方法的原因是什么”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“php无法在外部静态调用类和方法的原因是什么”吧!什么是静态调用?在PHP开发中...
    99+
    2023-07-05
  • C++调用动态库和Python调用C++动态库的方法是什么
    这篇文章主要介绍“C++调用动态库和Python调用C++动态库的方法是什么”,在日常操作中,相信很多人在C++调用动态库和Python调用C++动态库的方法是什么问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答...
    99+
    2023-07-05
  • android 调用JNI SO动态库的方法
    总结一下: android 调用JNI 分为静态调用与动态调用(不论动态还是静态前提都是NDK环境已经配置好的前提下) 一、静态主要就是将c(.c)或者c++(cpp)的源文件直接加...
    99+
    2022-11-12
  • linux中使用boost.python调用c++动态库的方法
    前言 最近开始使用 robot framework 测试c++的动态库,robot framework 是跑在 windows 上面,c++动态库是跑在远程linux主机上面。测试办法是让 robot framework...
    99+
    2022-06-04
    linux使用python linux boost.python linux调用动态库
  • Golang如何调用windows下的dll动态库中的函数
    使用syscall调用 package main import ( "fmt" "syscall" "time" "unsafe" ) const...
    99+
    2022-11-12
  • 如何在C++类的外部调用类的私有方法
    目录前言问题技术准备1. pointers to member functions2. The explicit template instantiation3. Passing a...
    99+
    2022-11-13
  • python嵌套函数使用外部函数变量的方法(Python2和Python3)
    python嵌套函数使用外部函数变量的方法,Python2和Python3均可使用 python3 def b(): b = 1 def bchange(): nonlocal b b...
    99+
    2022-06-04
    函数 嵌套 变量
  • python中调用函数的方法
    python中调用函数的方法:在python项目中新建一个函数,直接可以使用函数名加括号进行调用函数即可。具体步骤如下:打开python编辑器,新建一个py文档。在py文档使用def来新建一个函数。最后在通过函数名加括号直接进行调用该函数即...
    99+
    2022-10-10
  • Qt之调用C#的动态库的解决方法
    环境:VS2019+Qt5.12 1. CLR库安装         首先,如果你VS2019没有安装...
    99+
    2022-11-12
  • Go语言中调用外部命令的方法总结
    目录引子运行命令显示输出显示到标准输出输出到文件发送到网络保存到内存对象中输出到多个目的地运行命令,获取输出分别获取标准输出和标准错误标准输入环境变量检查命令是否存在封装总结引子 在...
    99+
    2022-11-11
  • android调用JNI SO动态库的方法是什么
    这篇文章主要介绍“android调用JNI SO动态库的方法是什么”,在日常操作中,相信很多人在android调用JNI SO动态库的方法是什么问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”android调用...
    99+
    2023-06-25
  • SQLServer2005数据库游标调用函数的使用方法
    这篇文章主要讲解了“SQLServer2005数据库游标调用函数的使用方法”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“SQLServer2005数据库游标...
    99+
    2022-10-18
  • Rust应用调用C语言动态库的操作方法
    目录外部功能接口FFIUDP套接字的读超时Rust调用C语言动态库中的函数避免重复造轮子,使用Rust官方C语言库外部功能接口FFI 虽然高级(脚本)编程语言的功能丰富,表达能力强,...
    99+
    2023-01-29
    Rust调用C语言动态库 Rust C语言动态库
  • JavaScript中子函数访问外部变量的3种解决方法
    前言 我们在写web页面时,肯定会经常遇到下面这种情况: <body> <div class="btns-wrapper"></div>...
    99+
    2022-11-12
  • vue引用外部JS并调用JS文件中的方法实例
    目录VUE项目中引入JS文件的几种方法1.在index.html页面使用script标签引入2.在main.js中使用window.moduleName 使用3.手动添加export...
    99+
    2023-02-27
    vue引用外部js文件 vuecli引入外部js
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作