2501,编写dll
DLL
的优点
简单的说,dll
有以下几个优点
:
1
)节省内存
.同一个软件模块
,若是源码重用
,则会在不同可执行程序
中编译
,同时运行这些exe
时,会在内存
中重复加载
这些模块的二进制码
.
如果使用dll
,则只在内存
中加载一次
,所有使用该dll
的进程会共享此块内存
(当然,每个进程
会复制
一份的dll
中的全局变量
).
2
)不需编译
的软件系统升级
,若一个软件系统
使用了dll
,则改变该dll
(函数名
不变)时,系统升级
只需要切换此dll
即可,不需要重新编译整个系统
.
3
)多种语言可使用Dll
库,如用c编写的dll
可在vb
中调用.DLL
还做得很不够,因此在dll
的基础上发明了COM
技术,更好的解决
了一系列问题
.
最简单的dll
最简单的dll
并不比c的helloworld
难,只要一个DllMain
函数即可,包含objbase.h
头文件(支持COM
技术的一个头文件
).
若该头文件
名字难记,则用windows.h
也可以.源码如下:dll_nolib.cpp
#include <objbase.h>
#include <iostream.h>
BOOL APIENTRY DllMain(HANDLE hModule, DWORD dwReason, void* lpReserved)
{HANDLE g_hModule;switch(dwReason){case DLL_PROCESS_ATTACH:cout<<"Dll is attached!"<<endl;g_hModule = (HINSTANCE)hModule;break;case DLL_PROCESS_DETACH:cout<<"Dll is detached!"<<endl;g_hModule=NULL;break;}return true;
}
其中DllMain
是每个dll
的入口函数,如同c的主
函数一样.DllMain
带三个参数,hModule
表示本dll
的实例句柄
,dwReason
表示dll
当前所处的状态,如DLL_PROCESS_ATTACH
表示dll
刚刚被加载
到一个进程
中,DLL_PROCESS_DETACH
表示刚刚从一个进程
中卸载dll
.
当然还有表示加载到线程
中和从线程中卸载
的状态,这里省略.最后参数
是一个保存参数
.
如上,在一个进程
中加载dll
时,dll
打印"Dllisattached!"
语句;当从进程
中卸载dll
时,打印"Dllisdetached!"
语句.
编译dll
需要以下两条命令
:
cl /c dll_nolib.cpp
这条命令
会按obj
文件编译cpp
,若不使用/c参数
,则cl
还会继续链接obj
为exe
,但是这里是一个dll
,没有主
函数,因此会报错
.不要紧,继续使用链接命令
.
Link /dll dll_nolib.obj
这条命令
会生成dll_nolib.dll
.
加载DLL
(显式调用)
一般有两个方式
使用dll
,显式调用和隐式调用
.这里首先介绍显式调用
.编写一个客户程序
:dll_nolib_client.cpp
#include <windows.h>
#include <iostream.h>
int main(void)
{//加载的`dll`HINSTANCE hinst=::LoadLibrary("dll_nolib.dll");if (NULL != hinst){cout<<"dll loaded!"<<endl;}return 0;
}
注意,调用dll
使用LoadLibrary
函数,它的参数
就是dll
的路径和名字
,返回值
是dll
的句柄
.使用如下命令
编译链接客户:
Cl dll_nolib_client.cpp
并执行dll_nolib_client.exe
,得到如下结果:
Dllisattached!
dllloaded!
Dllisdetached!
以上结果表明客户已加载dll
.但是这样仅可在内存加载dll
,不能找到dll
中的函数
.
使用dumpbin
命令查看DLL
中的函数
Dumpbin
命令可查看一个dll
中的输出函数符号名
,输入如下命令
:
Dumpbin -exports dll_nolib.dll
查看
发现dll_nolib.dll
并没有输出函数
.
如何在dll
中定义输出函数
总体来说
有两个方法
,一个
是添加
一个def
定义文件,在此文件
中定义dll
中要输出的函数
;第二个
是在源码
中,待输出的函数
前加上__declspec(dllexport)
关键字.
Def
文件
首先写一个带输出函数
的dll
,源码如下:dll_def.cpp
#include <objbase.h>
#include <iostream.h>
void FuncInDll (void)
{cout<<"FuncInDll is called!"<<endl;
}
BOOL APIENTRY DllMain(HANDLE hModule, DWORD dwReason, void* lpReserved)
{HANDLE g_hModule;switch(dwReason){case DLL_PROCESS_ATTACH:g_hModule = (HINSTANCE)hModule;break;case DLL_PROCESS_DETACH:g_hModule=NULL;break;}return TRUE;
}
该dll
的def
文件如下:dll_def.def
;
//`dll_def`模块定义文件
;
LIBRARY dll_def.dll
DESCRIPTION '(c)2007-2009 Wang Xuebin'
EXPORTSFuncInDll @1 PRIVATE
def
的语法很简单
,首先是库
关键字,指定dll
的名字;然后一个可选
的描述
关键字.
最后是导出
关键字,后面写上dll
中所有要输出的函数名或变量名
,然后接上@及依次编号的数字
(从1到N
),最后接上修饰符
.
用如下命令
编译链接带def
文件的dll
:
Cl /c dll_def.cpp
Link /dll dll_def.obj /def:dll_def.def
再调用dumpbin
查看生成的dll_def.dll
:
Dumpbin -exports dll_def.dll
得到结果
.
观察这一行
.
1000001000FuncInDll
会发现该dll
输出了FuncInDll
函数.
显式调用DLL
中的函数
写一个dll_def.dll
的客户程序:dll_def_client.cpp
#include <windows.h>
#include <iostream.h>
int main(void)
{//定义一个函数指针typedef void (* DLLWITHLIB )(void);//定义一个函数指针变量DLLWITHLIB pfFuncInDll = NULL;//加载`dll`HINSTANCE hinst=::LoadLibrary("dll_def.dll");if (NULL != hinst){cout<<"dll loaded!"<<endl;}//找到`dll`的`FuncInDll`函数pfFuncInDll = (DLLWITHLIB)GetProcAddress(hinst, "FuncInDll");//调用`dll`里的函数if (NULL != pfFuncInDll){(*pfFuncInDll)();}return 0;
}
有两个地方
值得注意,第一是定义和使用函数指针
;第二是使用GetProcAddress
,来查找dll
中的函数地址
.
第一个参数
是DLL
的句柄
,即LoadLibrary
返回的句柄
,第二个参数
是dll
中的函数名
,即dumpbin
中输出的函数名
.
注意,这里的函数名
指的是编译后的函数名
,不一定等于dll
源码中的函数名
.
编译链接
该客户程序
,执行得到:
dllloaded!
FuncInDlliscalled!
即客户成功调用
了dll
中的FuncInDll
函数.
__declspec(dllexport)
为每个dll
写def
显得很麻烦,当前def
使用已比较少了,更多的是在源码
中,使用__declspec(dllexport)
定义dll
的输出函数
.
Dll
写法同上
,去掉def
文件,并在每个要输出的函数
前面加上__declspec(dllexport)
声明,如:
__declspec(dllexport) void FuncInDll (void)
这里提供一个dll
的dll_withlib.cpp
源程序,然后编译链接
.链接
时不需要指定/DEF
:参数,直接加/DLL
参数即可,
Cl /c dll_withlib.cpp
Link /dll dll_withlib.obj
然后使用dumpbin
命令查看,得到:
1 0 00001000 FuncInDll@@YAXXZ
可知编译
后的函数名
为FuncInDll@@YAXXZ
.
可用extern"C"
指令来命令c++
编译器按c编译器
的方式来命名该函数
.如下:
extern "C" __declspec(dllexport) void FuncInDll (void)
dumpbin
命令结果:
1000001000 FuncInDll
这样,显式调用
时只需查找函数名
为FuncInDll
的函数
即可成功.
隐式调用DLL
显式调用
显得非常复杂,每次都要LoadLibrary
,并且每个函数
都必须使用GetProcAddress
来得到函数指针
,对大量使用dll
函数的客户
是个困扰.
而隐式调用
可像使用c函数库
一样使用dll
中的函数,非常方便快捷
.
下面是一个隐式调用
的示例:dll
包含两个文件dll_withlibAndH.cpp
和dll_withlibAndH.h
.
代码如下:dll_withlibAndH.h
extern "C" __declspec(dllexport) void FuncInDll (void);
//dll_withlibAndH.cpp
#include <objbase.h>
#include <iostream.h>
#include "dll_withLibAndH.h"//看到没有,这就是增加的头文件
extern "C" __declspec(dllexport) void FuncInDll (void)
{cout<<"FuncInDll is called!"<<endl;
}
BOOL APIENTRY DllMain(HANDLE hModule, DWORD dwReason, void* lpReserved)
{HANDLE g_hModule;switch(dwReason){case DLL_PROCESS_ATTACH:g_hModule = (HINSTANCE)hModule;break;case DLL_PROCESS_DETACH:g_hModule=NULL;break;}return TRUE;
}
编译链接命令
:
Cl /c dll_withlibAndH.cpp
Link /dll dll_withlibAndH.obj
在隐式调用
时,需要在客户
中引入头文件
,并在链接
时指明dll
对应的lib
文件(dll
只要有函数输出
,则链接
时会产生一个与dll
同名的lib
文件)位置和名
.
然后如同调用api
函数库中的函数
一样调用dll
中的函数,不需要显式
的LoadLibrary
和GetProcAddress
.使用最方便
.
客户代码
如下:dll_withlibAndH_client.cpp
#include "dll_withLibAndH.h"
//注意路径,加载`dll`的或`项目`|设置|`链接`设置里
#pragma comment(lib,"dll_withLibAndH.lib")
int main(void)
{FuncInDll();//只要这样就可调用`dll`里的函数了return 0;
}
__declspec(dllexport)
和__declspec(dllimport
)配对使用
事实上不使用extern"C"
是可行
的,这时会按c++
的符号串编译函数
,如(FuncInDll@@YAXH@Z,FuncInDll@@YAXXZ
),当客户
也是c++
时,也能正确
的隐式调用
.
这时要考虑一种情况
:若DLL1.CPP
是源,DLL2.CPP
使用了DLL1
中的函数
,但同时DLL2
也是一个DLL
,也要输出
一些函数供Client.CPP
使用.
则在DLL2
中如何声明
所有的,既包含了从DLL1
中引入的函数
,还包括自己要输出的函数
的函数.此时就需要
同时使用__declspec(dllexport)
和__declspec(dllimport)
了.
前者用来装饰
本dll
中的输出函数
,后者用来装饰
从其它dll
中引入的函数
.
所有的源码
包括DLL1.H,DLL1.CPP,DLL2.H,DLL2.CPP,Client.cpp
.
值得关注的是DLL1
和DLL2
中都使用的一个编码方法
,见DLL2.H
#ifdef DLL_DLL2_EXPORTS
#define DLL_DLL2_API __declspec(dllexport)
#else
#define DLL_DLL2_API __declspec(dllimport)
#endif
DLL_DLL2_API void FuncInDll2(void);
DLL_DLL2_API void FuncInDll2(int);
在头文件
中这样定义DLL_DLL2_EXPORTS
和DLL_DLL2_API
宏,可确保DLL
端的函数用__declspec(dllexport)
装饰,而客户的函数
用__declspec(dllimport)
装饰.
当然,记得在编译dll
时加上参数/D "DLL_DLL2_EXPORTS"
,或干脆就在dll
的cpp
文件第一行
加上#define DLL_DLL2_EXPORTS
.
DLL
中的全局变量和对象
解决了重载函数
的问题,则dll
中的全局变量和对象
都不是问题了,只是有一点语法
注意.如源码所示:dll_object.h
#ifdef DLL_OBJECT_EXPORTS
#define DLL_OBJECT_API __declspec(dllexport)
#else
#define DLL_OBJECT_API __declspec(dllimport)
#endif
DLL_OBJECT_API void FuncInDll(void);
extern DLL_OBJECT_API int g_nDll;
class DLL_OBJECT_API CDll_Object {
public:CDll_Object(void);show(void);//`待办`:在此处添加你的方法.
};
Cpp
文件dll_object.cpp
如下:
#define DLL_OBJECT_EXPORTS
#include <objbase.h>
#include <iostream.h>
#include "dll_object.h"
DLL_OBJECT_API void FuncInDll(void)
{cout<<"FuncInDll is called!"<<endl;
}
DLL_OBJECT_API int g_nDll = 9;
CDll_Object::CDll_Object()
{cout<<"ctor of CDll_Object"<<endl;
}
CDll_Object::show()
{cout<<"function show in class CDll_Object"<<endl;
}
BOOL APIENTRY DllMain(HANDLE hModule, DWORD dwReason, void* lpReserved)
{HANDLE g_hModule;switch(dwReason){case DLL_PROCESS_ATTACH:g_hModule = (HINSTANCE)hModule;break;case DLL_PROCESS_DETACH:g_hModule=NULL;break;}return TRUE;
}
编译链接
完后Dumpbin
一下,可见输出了5个符号
:
1 0 00001040 0CDll_Object@@QAE@XZ2 1 00001000 4CDll_Object@@QAEAAV0@ABV0@@Z3 2 00001020 FuncInDll@@YAXXZ4 3 00008040 g_nDll@@3HA5 4 00001069 show@CDll_Object@@QAEHXZ
它们分别代表CDll_Object
类,类的构造器
,FuncInDll
函数,g_nDll
全局变量和类的显示
成员函数.下面是客户代码
:dll_object_client.cpp
#include "dll_object.h"
#include <iostream.h>
//注意路径,加载`dll`的或`项目`|设置|`链接`设置里
#pragma comment(lib,"dll_object.lib")
int main(void)
{cout<<"call dll"<<endl;cout<<"call function in dll"<<endl;FuncInDll();//只要这样就可调用`dll`里的函数了cout<<"global var in dll g_nDll ="<<g_nDll<<endl;cout<<"call member function of class CDll_Object in dll"<<endl;CDll_Object obj;obj.show();return 0;
}
运行该客户可见:
calldll
callfunctionindll
FuncInDlliscalled!
globalvarindllg_nDll=9
callmemberfunctionofclassCDll_Objectindll
ctorofCDll_Object
functionshowinclassCDll_Object
可知,客户
成功的访问了dll
中的全局变量
,并创建了dll
中定义的C++
对象,还调用
了该对象的成员函数
.
相关文章:
2501,编写dll
DLL的优点 简单的说,dll有以下几个优点: 1)节省内存.同一个软件模块,若是源码重用,则会在不同可执行程序中编译,同时运行这些exe时,会在内存中重复加载这些模块的二进制码. 如果使用dll,则只在内存中加载一次,所有使用该dll的进程会共享此块内存(当然,每个进程会复制一份的d…...
【router路由的配置】
router路由的配置 App.vuerouter在main.ts引入插件 App.vue <template><RouterView /> </template><script setup lang"ts"></script><style scoped lang"scss"></style>router import { createRouter, creat…...
算法基础——一致性
引入 最早研究一致性的场景既不是大数据领域,也不是分布式系统,而是多路处理器。 可以将多路处理器理解为单机计算机系统内部的分布式场景,它有多个执行单元,每一个执行单元都有自己的存储(缓存),一个执行单元修改了…...

刷题记录 动态规划-6: 62. 不同路径
题目:62. 不同路径 难度:中等 一个机器人位于一个 m x n 网格的左上角 (起始点在下图中标记为 “Start” )。 机器人每次只能向下或者向右移动一步。机器人试图达到网格的右下角(在下图中标记为 “Finish” &#x…...
docker直接运行arm下的docker
运行环境是树莓派A 处理器是 arm32v6 安装了docker,运行lamp 编译安装php的时候发现要按天来算,于是用电脑vm下的Ubuntu系统运行arm的docker 然后打包到a直接导入运行就可以了 第一种方法 sudo apt install qemu-user-static 导入直接运行就可以了…...

014-STM32单片机实现矩阵薄膜键盘设计
1.功能说明 本设计主要是利用STM32驱动矩阵薄膜键盘,当按下按键后OLED显示屏上会对应显示当前的按键键值,可以将此设计扩展做成电子秤、超市收银机、计算器等需要多个按键操作的单片机应用。 2.硬件接线 模块管脚STM32单片机管脚矩阵键盘行1PA0矩阵键盘…...

Sentinel 断路器在Spring Cloud使用
文章目录 Sentinel 介绍同类对比微服务雪崩问题问题原因问题解决方案请求限流线程隔离失败处理服务熔断解决雪崩问题的常见方案有哪些? Sentineldocker 安装账号/ 密码项目导入簇点链路请求限流线程隔离Fallback服务掉线时的处理流程 服务熔断 Sentinel 介绍 随着微…...
[内网安全] 内网渗透 - 学习手册
这是一篇专栏的目录文档,方便读者系统性的学习,笔者后续会持续更新文档内容。 如果没有特殊情况的话,大概是一天两篇的速度。(实验多或者节假日,可能会放缓) 笔者也是一边学习一边记录笔记,如果…...
算法总结-二分查找
文章目录 1.搜索插入位置1.答案2.思路 2.搜索二维矩阵1.答案2.思路 3.寻找峰值1.答案2.思路 4.搜索旋转排序数组1.答案2.思路 5.在排序数组中查找元素的第一个和最后一个位置1.答案2.思路 6.寻找旋转排序数组中的最小值1.答案2.思路 1.搜索插入位置 1.答案 package com.sunxi…...

基于python的Kimi AI 聊天应用
因为这几天deepseek有点状况,导致apikey一直生成不了,用kimi练练手。这是一个基于 Moonshot AI 的 Kimi 接口开发的聊天应用程序,使用 Python Tkinter 构建图形界面。 项目结构 项目由三个主要Python文件组成: 1. main_kimi.py…...
动手学深度学习-3.2 线性回归的从0开始
以下是代码的逐段解析及其实际作用: 1. 环境设置与库导入 %matplotlib inline import random import torch from d2l import torch as d2l作用: %matplotlib inline:在 Jupyter Notebook 中内嵌显示 matplotlib 图形。random:生成…...
Spring 面试题【每日20道】【其二】
1、Spring MVC 具体的工作原理? 中等 Spring MVC 是 Spring 框架的一部分,专门用于构建基于Java的Web应用程序。它采用模型-视图-控制器(MVC)架构模式,有助于分离应用程序的不同方面,如输入逻辑、业务逻辑…...

嵌入式八股文面试题(一)C语言部分
1. 变量/函数的声明和定义的区别? (1)变量 定义不仅告知编译器变量的类型和名字,还会分配内存空间。 int x 10; // 定义并初始化x int x; //同样是定义 声明只是告诉编译器变量的名字和类型,但并不为它分配内存空间…...

Vue06
目录 一、声明式导航-导航链接 1.需求 2.解决方案 3.通过router-link自带的两个样式进行高亮 二、声明式导航的两个类名 1.router-link-active 2.router-link-exact-active 三、声明式导航-自定义类名(了解) 1.问题 2.解决方案 3.代码演示 四…...

deepseek-r1模型本地win10部署
转载自大佬:高效快速教你deepseek如何进行本地部署并且可视化对话 deepseek 如果安装遇到这个问题 Error: Post “http://127.0.0.1:11434/api/show”: read tcp 127. 用管理员cmd打开 接着再去切换盘符d: cd 文件夹 重新下载模型:ollama run deepseek…...
自定义数据集 使用scikit-learn中SVM的包实现SVM分类
生成自定义数据集 生成一个简单的二维数据集,包含两类数据点,分别用不同的标签表示。 import numpy as np import matplotlib.pyplot as plt# 生成数据 np.random.seed(42) X np.r_[np.random.randn(100, 2) - [2, 2], np.random.randn(100, 2) [2, …...
pandas的melt方法使用
Pandas 的 melt 方法用于将宽格式(wide format)的 DataFrame 转换为长格式(long format)的 DataFrame。这种转换在数据处理和可视化中非常有用,尤其是在处理多列数据时。 宽格式 vs 长格式 宽格式(Wide F…...

一文讲解Spring中应用的设计模式
我们都知道Spring 框架中用了蛮多设计模式的: 工厂模式呢,就是用来创建对象的,把对象的创建和使用分开,这样代码更灵活。代理模式呢,是用一个代理对象来控制对真实对象的访问,可以在访问前后做一些处理。单…...

Linux的基本指令(下)
1.find指令 Linux下find命令在⽬录结构中搜索⽂件,并执⾏指定的操作。 Linux下find命令提供了相当多的查找条件,功能很强⼤。由于find具有强⼤的功能,所以它的选项也很多,其中⼤部分选项都值得我们花时间来了解⼀下。 即使系统中含…...

HAO的Graham学习笔记
前置知识:凸包 摘录oiwiki 在平面上能包含所有给定点的最小凸多边形叫做凸包。 其定义为:对于给定集合 X,所有包含 X 的凸集的交集 S 被称为 X 的 凸包。 说人话就是用一个橡皮筋包含住所有给定点的形态 如图: 正题:…...
零门槛NAS搭建:WinNAS如何让普通电脑秒变私有云?
一、核心优势:专为Windows用户设计的极简NAS WinNAS由深圳耘想存储科技开发,是一款收费低廉但功能全面的Windows NAS工具,主打“无学习成本部署” 。与其他NAS软件相比,其优势在于: 无需硬件改造:将任意W…...
Cursor实现用excel数据填充word模版的方法
cursor主页:https://www.cursor.com/ 任务目标:把excel格式的数据里的单元格,按照某一个固定模版填充到word中 文章目录 注意事项逐步生成程序1. 确定格式2. 调试程序 注意事项 直接给一个excel文件和最终呈现的word文件的示例,…...
ssc377d修改flash分区大小
1、flash的分区默认分配16M、 / # df -h Filesystem Size Used Available Use% Mounted on /dev/root 1.9M 1.9M 0 100% / /dev/mtdblock4 3.0M...

ESP32读取DHT11温湿度数据
芯片:ESP32 环境:Arduino 一、安装DHT11传感器库 红框的库,别安装错了 二、代码 注意,DATA口要连接在D15上 #include "DHT.h" // 包含DHT库#define DHTPIN 15 // 定义DHT11数据引脚连接到ESP32的GPIO15 #define D…...

汽车生产虚拟实训中的技能提升与生产优化
在制造业蓬勃发展的大背景下,虚拟教学实训宛如一颗璀璨的新星,正发挥着不可或缺且日益凸显的关键作用,源源不断地为企业的稳健前行与创新发展注入磅礴强大的动力。就以汽车制造企业这一极具代表性的行业主体为例,汽车生产线上各类…...
【android bluetooth 框架分析 04】【bt-framework 层详解 1】【BluetoothProperties介绍】
1. BluetoothProperties介绍 libsysprop/srcs/android/sysprop/BluetoothProperties.sysprop BluetoothProperties.sysprop 是 Android AOSP 中的一种 系统属性定义文件(System Property Definition File),用于声明和管理 Bluetooth 模块相…...

Map相关知识
数据结构 二叉树 二叉树,顾名思义,每个节点最多有两个“叉”,也就是两个子节点,分别是左子 节点和右子节点。不过,二叉树并不要求每个节点都有两个子节点,有的节点只 有左子节点,有的节点只有…...
Java线上CPU飙高问题排查全指南
一、引言 在Java应用的线上运行环境中,CPU飙高是一个常见且棘手的性能问题。当系统出现CPU飙高时,通常会导致应用响应缓慢,甚至服务不可用,严重影响用户体验和业务运行。因此,掌握一套科学有效的CPU飙高问题排查方法&…...

华硕a豆14 Air香氛版,美学与科技的馨香融合
在快节奏的现代生活中,我们渴望一个能激发创想、愉悦感官的工作与生活伙伴,它不仅是冰冷的科技工具,更能触动我们内心深处的细腻情感。正是在这样的期许下,华硕a豆14 Air香氛版翩然而至,它以一种前所未有的方式&#x…...
音视频——I2S 协议详解
I2S 协议详解 I2S (Inter-IC Sound) 协议是一种串行总线协议,专门用于在数字音频设备之间传输数字音频数据。它由飞利浦(Philips)公司开发,以其简单、高效和广泛的兼容性而闻名。 1. 信号线 I2S 协议通常使用三根或四根信号线&a…...