Linux下c语言实现动态库的动态调用
在Linux操作系统下,有时候需要在不重新编译程序的情况下,运行时动态地加载库,这时可以通过Linux操作系统提供的API可以实现,涉及到的API主要有dlopen、dlsym和dlclose。使用时,需要加上头文件#include <dlfcn.h> 。
dlopen介绍:打开一个动态链接库 ,函数定义如下:
void * dlopen( const char * pathname, int mode );
函数功能描述:在dlopen的()函数以指定模式打开指定的动态连接库文件,并返回一个句柄给调用进程。
参数说明:
pathname:动态库的名称,需要带上路径。
mode:分为这几种
RTLD_LAZY 暂缓决定,等有需要时再解出符号
RTLD_NOW 立即决定,返回前解除所有未决定的符号。
RTLD_LOCAL
RTLD_GLOBAL 允许导出符号
RTLD_GROUP
RTLD_WORLD
返回值说明:
打开错误返回空指针NULL ,若成功,返回库引用
dlsym介绍:
该函数根据动态链接库操作句柄与符号,返回符号对应的地址。
函数定义如下:
void*dlsym(void* handle,const char* symbol);
函数说明:
dlsym根据动态链接库操作句柄和符号,返回符号对应的地址。使用这个函数不但可以获取函数地址,也可以获取变量地址。
参数说明:
handle:打开库文件之后的句柄。
symbol:需要从库文件查找的符号。
dlclose
dlclose用于关闭指定句柄的动态链接库,只有当此动态链接库的使用计数为0时,才会真正被系统卸载。
实验一 获取函数地址
在linux下创建一个test的工程目录。
- 工程目录下创建一个名为lib1.c的文件,写入如下内容:
#include<stdio.h> #include<stdlib.h> #include <stdarg.h>void LOG(const char *format, ...) {va_list argptr;char buffer[2048];va_start(argptr,format);vsprintf(buffer,format,argptr);va_end(argptr);printf("%s\n", buffer); }void lib_function_1(void) {LOG("call %s!!!", __func__); }void lib_function_2(void) {LOG("call %s!!!", __func__); }
- 创建一个main.c文件,内容如下所示:
#include<stdio.h> #include<stdlib.h> #include <dlfcn.h>void dynamic_lib_test() {void (*fun)();void *hander = NULL;hander = dlopen("./libshare.so", RTLD_NOW);if(hander == NULL) {printf("can not find dlib\n");return;}fun = (void(*)())dlsym(hander, "lib_function_1");if(fun==NULL) {printf("can't find function\n");}fun();fun = (void(*)())dlsym(hander, "lib_function_2");if(fun==NULL) {printf("can't find function\n");}fun();dlclose(hander); }int main(int argc, char *argv[]) {dynamic_lib_test();return 0; }
- 创建一个Makefile,生成一个动态库以及可执行文件,内容如下所示:
CPROG = test BIN = $(CPROG) CC= gcc OBJS=main.o lib1.oLDFLAGS += -ldlall: $(BIN) clean:rm -f $(OBJS) $(BIN) $(BIN): $(OBJS)$(CC) -g -fPIC -shared lib1.c -o libshare.so$(CC) -o $(BIN) $(OBJS) $(CFLAGS) $(LDFLAGS) $(CFLAGS_EXTRA)
- 编译该工程的代码
执行命令:make clean;make
最终会在该工程生成一个libshare.so动态库文件以及test的可执行文件。
- 测试验证在该工程下执行./test,便可以观察到最终结果,如下图所示:
实验二 获取全局变量地址
- 在当前目录下新增lib2.c,写入如下内容:
#include<stdio.h> #include<stdlib.h> #include<string.h> #include<stdarg.h>//声明一个结构体 typedef struct test_s {char *test_name;void (*test_func)();void (*test_set_buf)(char *val);char *(*test_get_buf)(); }test_t;char testbuf[128]="123456";void Log(const char *format, ...) {va_list argptr;char buffer[2048];va_start(argptr,format);vsprintf(buffer,format,argptr);va_end(argptr);printf("%s\n", buffer); } void test_function() {Log("function %s call!!!", __func__); }void test_set_buf(char *val) {strcpy(testbuf, val); }char *test_get_buf() {return testbuf; } //定义一个全局变量 test_t test = {.test_name = "TestName",.test_func = test_function,.test_get_buf = test_get_buf,.test_set_buf = test_set_buf, };
- 编辑main.c,内容如下:
#include <string.h>void dynamic_lib_test_1() {void (*fun)();void *hander = NULL;hander = dlopen("./libshare.so", RTLD_NOW);if(hander == NULL) {printf("can not find dlib\n");return;}fun = (void(*)())dlsym(hander, "lib_function_1");if(fun==NULL) {printf("can't find function\n");}fun();fun = (void(*)())dlsym(hander, "lib_function_2");if(fun==NULL) {printf("can't find function\n");}fun();dlclose(hander); }typedef struct test_s {char *test_name;void (*test_func)();void (*test_set_buf)(char *val);char *(*test_get_buf)(); }test_t;void dynamic_lib_test_2() {void *hander = NULL;hander = dlopen("./libshare.so", RTLD_NOW);if(hander == NULL) {printf("can not find dlib\n");return;}test_t *t = (test_t *)dlsym(hander, "test");if(t==NULL) {printf("can't find function\n");return;}printf("name:%s, buf:%s\n", t->test_name, t->test_get_buf());t->test_func();t->test_set_buf("hello world!!!");printf("name:%s, buf:%s\n", t->test_name, t->test_get_buf());dlclose(hander); }int main(int argc, char *argv[]) {dynamic_lib_test_1();dynamic_lib_test_2();return 0; }
- 编辑Makefile,主要是添加lib2.c的编译
CPROG = test BIN = $(CPROG) CC= gcc OBJS=main.o lib1.o LDFLAGS += -ldlall: $(BIN) clean:rm -f $(OBJS) $(BIN) $(BIN): $(OBJS)$(CC) -g -fPIC -shared lib1.c lib2.c -o libshare.so$(CC) -o $(BIN) $(OBJS) $(CFLAGS) $(LDFLAGS) $(CFLAGS_EXTRA)
4.执行make clean;make重新编译这个工程
5.测试验证
dlsym找到全局结构体test后,可以直接用这个全局结构体指针来使用库里面的函数了。
总结:
通过dlopen打开动态库的方式,允许在运行时动态地加载库,这可以让你在不重新编译程序的情况下,添加或修改库中的函数,同时也为程序提供了更大的灵活性。dlsym允许程序在运行时查找库中的符号(通常是函数或变量的名称)。这使得程序可以在运行时决定调用哪个版本的函数,或者根据需要选择不同的实现。由于只有当程序实际需要时才加载库,因此可以节省内存。如果多个程序共享同一个库,那么这个库只需要在内存中加载一次。使用动态链接,你可以更容易地控制库的版本。
相关文章:
Linux下c语言实现动态库的动态调用
在Linux操作系统下,有时候需要在不重新编译程序的情况下,运行时动态地加载库,这时可以通过Linux操作系统提供的API可以实现,涉及到的API主要有dlopen、dlsym和dlclose。使用时,需要加上头文件#include <dlfcn.h>…...
为什么MCU在ADC采样时IO口有毛刺?
大家在使用MCU内部ADC进行信号采样一个静态电压时,可能在IO口上看到这样的波形。这个时候大家一般会认识是信号源有问题,但仔细观察会发现这个毛刺的频率是和ADC触发频率一样的。 那么为什么MCU在ADC采样时IO口会出现毛刺呢?这个毛刺对结果有…...
C# 将 Word 转化分享为电子期刊
目录 需求 方案分析 相关库引入 关键代码 Word 转 Pdf Pdf 转批量 Jpeg Jpeg 转为电子书 实现效果演示 小结 需求 曾经的一个项目,要求实现制作电子期刊定期发送给企业进行阅读,基本的需求如下: 1、由编辑人员使用 Microsoft Word…...
网络世界的黑暗角落:常见漏洞攻防大揭秘
网络世界的黑暗角落:常见漏洞攻防大揭秘 今天带来了网站常见的漏洞总结,大家在自己的服务器上也需要好好进行防护,密码不要过于简单.不然非常容易遭到攻击,最终达到不可挽回的损失.很多黑客想网络乞丐一样将你服务器打宕机,然后要求你进行付费.不知道大家有没有遇到…...
通信领域发展方向
5G网络技术:随着5G网络的建设和商用推广,各家运营商、厂商和研究机构都在探索5G技术的应用场景和解决方案,如网络切片、毫米波通信、多用户MIMO等。 物联网技术:物联网技术已经成为通信行业的重点发展领域,包括传感器…...
21 3GPP中 5G NR高速列车通信标准化
文章目录 信道模型实验——物理层设计相关元素μ(与子载波间隔有关)设计参考信号(DMRS) 本文提出初始接入、移动性管理、线性小区设计等高层技术。描述3GPP采用HST场景的评估参数,阐释了HST应用的物理层技术,包括数字通信和参考信号设计,链路…...
【网络安全】-Linux操作系统—CentOS安装、配置
文章目录 准备工作下载CentOS创建启动盘确保硬件兼容 安装CentOS启动安装程序分区硬盘网络和主机名设置开始安装完成安装 初次登录和配置更新系统安装额外的软件仓库安装网络工具配置防火墙设置SELinux安装文本编辑器配置SSH服务 总结 CentOS是一个基于Red Hat Enterprise Linu…...
CCNP课程实验-OSPF-CFG
目录 实验条件网络拓朴需求 配置实现基础配置1. 配置所有设备的IP地址 实现目标1. 要求按照下列标准配置一个OSPF网络。 路由协议采用OSPF,进程ID为89 ,RID为loopback0地址。3. R4/R5/R6相连的三个站点链路OSPF网络类型配置成广播型,其中R5路…...
【Spring Security】打造安全无忧的Web应用--入门篇
🥳🥳Welcome Huihuis Code World ! !🥳🥳 接下来看看由辉辉所写的关于Spring Security的相关操作吧 目录 🥳🥳Welcome Huihuis Code World ! !🥳🥳 一.Spring Security是什么 1.概…...
【每日一题】【12.20】2828.判别首字母缩略词
🔥博客主页: A_SHOWY🎥系列专栏:力扣刷题总结录 数据结构 云计算 数字图像处理 力扣每日一题_ 1.题目链接 2828. 判别首字母缩略词https://leetcode.cn/problems/check-if-a-string-is-an-acronym-of-words/ 2.题目描述 今天…...
LabVIEW开发振动数据分析系统
LabVIEW开发振动数据分析系统 自动测试系统基于LabVIEW平台设计,采用了多种高级硬件设备。系统的硬件组成包括PCB振动加速度传感器,这是一种集成了传统压电加速度传感器和电荷放大器的先进设备,能够直接与采集仪器连接。此外,系统…...
去掉乘法运算的加法移位神经网络架构
[CVPR 2020] AdderNet: Do We Really Need Multiplications in Deep Learning? 代码:https://github.com/huawei-noah/AdderNet/tree/master 核心贡献 用filter与input feature之间的L1-范数距离作为“卷积层”的输出为了提升模型性能,提出全精度梯度…...
【TB作品】51单片机,具有报时报温功能的电子钟
2.具有报时报温功能的电子钟 一、功能要求: 1.显示室温。 2.具有实时时间显示。 3.具有实时年月日显示和校对功能。 4.具有整点语音播报时间和温度功能。 5.定闹功能,闹钟音乐可选。 6.操作简单、界面友好。 二、设计建议: 1.单片机自选(C51、STM32或其他单片机)。 2.时钟日历芯…...
了解C++工作机制
基于hello.cpp对C的运行进行一个初步认识,并介绍国外C大佬Cherno常用的项目结构和调试Tips C是如何工作的 C工作流程1.实用工程(project)结构(1)Microsoft Visual Studio2022新建项目后,自动生成的原始文件…...
力扣题目学习笔记(OC + Swift) 14. 最长公共前缀
14. 最长公共前缀 编写一个函数来查找字符串数组中的最长公共前缀。 如果不存在公共前缀,返回空字符串 “”。 方法一 竖向扫描法 个人感觉纵向扫描方式比较直观,符合人类理解方式,从前往后遍历所有字符串的每一列,比较相同列上的…...
WinSW设置应用程序开机启动
前言 由于使用windows自动的自启方法,不管是将程序启动服务放到开机自启文件夹中,还是创建任务计划程序,都没有很好的实现程序的开机自启效果,而WinSW很好的解决了这个问题。 下载 WinSW下载地址 注意:不同版本&#…...
Leetcode—96.不同的二叉搜索树【中等】
2023每日刷题(六十四) Leetcode—96.不同的二叉搜索树 算法思想 实现代码 class Solution { public:int numTrees(int n) {vector<int> G(n 1, 0);G[0] 1;G[1] 1;for(int i 2; i < n; i) {for(int j 1; j < i; j) {G[i] G[j - 1] * …...
正则表达式零宽断言
正则表达式零宽断言 工具类,正则表达式匹配文本内容正则表达式语法例子例子01零宽断言?< 不包含左边值? 不包含右边值例子 常用正则表达式校验数字的表达式校验字符的表达式 工具类,正则表达式匹配文本内容 /*** 正则表达式工具类*/ public class…...
uni-app学习记录
uni-app注意点记录 跳转到 tabBar 页面只能使用 switchTab 跳转路由API的目标页面必须是在pages.json里注册的vue页面。如果想打开web url,在App平台可以使用 plus.runtime.openURL或web-view组件;H5平台使用 window.open;小程序平台使用web…...
API资源对象StorageClass;Ceph存储;搭建Ceph集群;k8s使用ceph
API资源对象StorageClass;Ceph存储;搭建Ceph集群;k8s使用ceph API资源对象StorageClass SC的主要作用在于,自动创建PV,从而实现PVC按需自动绑定PV。 下面我们通过创建一个基于NFS的SC来演示SC的作用。 要想使用NFS的SC,还需要安装一个NFS…...
Databend 开源周报第 124 期
Databend 是一款现代云数仓。专为弹性和高效设计,为您的大规模分析需求保驾护航。自由且开源。即刻体验云服务:https://app.databend.cn 。 Whats On In Databend 探索 Databend 本周新进展,遇到更贴近你心意的 Databend 。 新增对 Delta 和…...
Arduino开发实例-液体流量测量
液体流量测量 文章目录 液体流量测量1、流量传感器介绍2、硬件准备及接线3、代码实现在本文中,将介绍如何流量传感器进行测量液体流量。 流量传感器用于测量液体流速。 市场上有不同类型的流量传感器,在本文中,我们将使用霍尔效应流量传感器。 这些类型的流量传感器是非侵入…...
【idea】解决sprintboot项目创建遇到的问题
目录 一、报错Plugin ‘org.springframework.boot:spring-boot-maven-plugin:‘ not found 二、报错java: 错误: 无效的源发行版:17 三、java: 无法访问org.springframework.web.bind.annotation.CrossOrigin 四、整合mybatis的时候,报java.lang.Ill…...
ADC芯片CS1237在电子秤方案的优势
随着科技的不断发展,电子秤已经成为我们日常生活中不可或缺的测量工具。为了满足用户对于高精度、高稳定性的需求,芯海ADC芯片CS1237应运而生,为电子秤方案带来了革命性的变革。 一、芯海ADC芯片CS1237介绍 芯海ADC芯片CS1237是一款高性能…...
Leetcode的AC指南 —— 哈希表:202. 快乐数
摘要: Leetcode的AC指南 —— 哈希表:202. 快乐数。题目介绍:编写一个算法来判断一个数 n 是不是快乐数。 文章目录 一、题目二、解析1、哈希表 一、题目 题目介绍:编写一个算法来判断一个数 n 是不是快乐数。 「快乐数」 定义为…...
机器学习 项目结构 数据预测 实验报告
需求: 我经过处理得到了测试值,然后进一步得到预测和真实值的比较,然后再把之前的所有相关的参数、评估指标、预测值、比较结果都存入excel,另外我还打算做测试报告模板,包括敏感性分析等。您建议我这些功能如何封装这些功能&…...
[Verilog] 设计方法和设计流程
主页: 元存储博客 文章目录 1. 设计方法2. 设计流程 3 Vivado软件设计流程总结 1. 设计方法 Verilog 的设计多采用自上而下的设计方法(top-down)。设计流程是指从一个项目开始从项目需求分析,架构设计,功能验证&#…...
C语言:指向数组的指针和指向数组元素的指针
相关阅读 C语言https://blog.csdn.net/weixin_45791458/category_12423166.html?spm1001.2014.3001.5482 指向数组的指针和指向数组元素的指针常常被混淆,或者笼统地被称为数组指针,但它们之间是有差别的,本文就将对此进行讨论。 下面的代码…...
SQL基础:SQL 介绍和数据库基础
SQL简介 常用的Java等语言是和计算机交流的工具,告诉计算机,让计算机做一些事。 和其类似,SQL是 Structured Query Language 的缩写,即结构化的查询语言,是和数据库交互的工具,即通过既定的一些格式&…...
SpringSecurity入门
前言 Spring Security是一个用于在Java应用程序中提供身份验证和授权功能的强大框架。它构建在Spring框架之上,为开发人员提供了一套灵活且全面的安全性服务,本篇将为大家带来Spring Security的详细介绍及入门 一.安全框架 在学习了解Spring Security之…...
网站注册页面怎么做数据验证码/网站免费推广网站
作为“软件定义汽车”的“先行者”,智能座舱已经成为全球汽车企业的“风向标”。 相比于自动驾驶在法律法规、技术路径等方面的不确定性,智能座舱的商业化落地容易推进。因此,作为智能座舱的核心部件,芯片市场率先迎来了新一轮的…...
wordpress 教育类主题/推广策划方案怎么做
摘要:CSS中的边框用户登录批改老师:西门大官人批改时间:2019-02-24 13:56:24老师总结:作业完成的不错,最好上传一下运行的效果图...
产品开发计划书/宁波seo怎么做推广渠道
源码路径:dangwei-90/DllLoadAndFree (github.com) DLL显示调用的加载和卸载顺序:[DLL] dll 的加载和卸载顺序 (显示调用)_二七-CSDN博客 dll 分为显示加载和隐式加载。 假设 program 加载 dllA , 而dllA 加载 dllB, 隐式加载顺序如下: 1.…...
做网站公司那家好/推广获客
pyramid setup参考http://docs.pylonsproject.org/en/latest/docs/pyramid.html今天是2012-03-12讲的都是安装好Python2.7,以及在ubuntu操作系统下面:1.首先装Pyramid,测试语句import setuptools,没有ImportError错误,…...
重庆网站建设注意事项/快排seo
接上一篇 WPF多进程UI探索(Like Chrome) 找到了相对较靠谱的跨进程传递WPFUI的方法,本篇将对WPF多进程UI框架进行设计。 功能性需求 一个主进程作为宿主,承载多个子进程的UI每个子进程相互独立,互不影响主进程和子进程…...