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还会继续链接objexe,但是这里是一个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;
}

dlldef文件如下: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)

为每个dlldef显得很麻烦,当前def使用已比较少了,更多的是在源码中,使用__declspec(dllexport)定义dll输出函数.

Dll写法同上,去掉def文件,并在每个要输出的函数前面加上__declspec(dllexport)声明,如:

__declspec(dllexport) void FuncInDll (void)

这里提供一个dlldll_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.cppdll_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中的函数,不需要显式LoadLibraryGetProcAddress.使用最方便.

客户代码如下: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.
值得关注的是DLL1DLL2中都使用的一个编码方法,见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_EXPORTSDLL_DLL2_API宏,可确保DLL端的函数用__declspec(dllexport)装饰,而客户的函数__declspec(dllimport)装饰.

当然,记得在编译dll时加上参数/D "DLL_DLL2_EXPORTS",或干脆就在dllcpp文件第一行加上#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++对象,还调用该对象的成员函数.

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/10980.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Spring Boot + Facade Pattern : 通过统一接口简化多模块业务

文章目录 Pre概述在编程中&#xff0c;外观模式是如何工作的&#xff1f;外观设计模式 UML 类图外观类和子系统的关系优点案例外观模式在复杂业务中的应用实战运用1. 项目搭建与基础配置2. 构建子系统组件航班服务酒店服务旅游套餐服务 3. 创建外观类4. 在 Controller 中使用外…

八、Spring Boot 日志详解

目录 一、日志的用途 二、日志使用 2.1 打印日志 2.1.1 在程序中获取日志对象 2.1.2 使用日志对象打印日志 2.2、日志框架介绍 2.2.1 门面模式(外观模式) 2.2.2 门面模式的实现 2.2.3 SLF4J 框架介绍 2.3 日志格式的说明 2.4 日志级别 2.4.1 日志级别的分类 2.4.2…

创建前端项目的方法

目录 一、创建前端项目的方法 1.前提&#xff1a;安装Vue CLI 2.方式一&#xff1a;vue create项目名称 3.方式二&#xff1a;vue ui 二、Vue项目结构 三、修改Vue项目端口号的方法 一、创建前端项目的方法 1.前提&#xff1a;安装Vue CLI npm i vue/cli -g 2.方式一&…

(leetcode 213 打家劫舍ii)

代码随想录&#xff1a; 将一个线性数组换成两个线性数组&#xff08;去掉头&#xff0c;去掉尾&#xff09; 分别求两个线性数组的最大值 最后求这两个数组的最大值 代码随想录视频 #include<iostream> #include<vector> #include<algorithm> //nums:2,…

【Python】第七弹---Python基础进阶:深入字典操作与文件处理技巧

✨个人主页&#xff1a; 熬夜学编程的小林 &#x1f497;系列专栏&#xff1a; 【C语言详解】 【数据结构详解】【C详解】【Linux系统编程】【MySQL】【Python】 目录 1、字典 1.1、字典是什么 1.2、创建字典 1.3、查找 key 1.4、新增/修改元素 1.5、删除元素 1.6、遍历…

本地部署DeepSeek开源多模态大模型Janus-Pro-7B实操

本地部署DeepSeek开源多模态大模型Janus-Pro-7B实操 Janus-Pro-7B介绍 Janus-Pro-7B 是由 DeepSeek 开发的多模态 AI 模型&#xff0c;它在理解和生成方面取得了显著的进步。这意味着它不仅可以处理文本&#xff0c;还可以处理图像等其他模态的信息。 模型主要特点:Permalink…

BW AO/工作簿权限配置

场景&#xff1a; 按事业部配置工作簿权限&#xff1b; 1、创建用户 事务码&#xff1a;SU01&#xff0c;用户主数据的维护&#xff0c;可以创建、修改、删除、锁定、解锁、修改密码等 用户设置详情页 2、创建权限角色 用户的权限菜单是通过权限角色分配来实现的 2.1、自定…

jstat命令详解

jstat 用于监视虚拟机运行时状态信息的命令&#xff0c;它可以显示出虚拟机进程中的类装载、内存、垃圾收集、JIT 编译等运行数据。 命令的使用格式如下。 jstat [option] LVMID [interval] [count]各个参数详解&#xff1a; option&#xff1a;操作参数LVMID&#xff1a;本…

3.Spring-事务

一、隔离级别&#xff1a; 脏读&#xff1a; 一个事务访问到另外一个事务未提交的数据。 不可重复读&#xff1a; 事务内多次查询相同条件返回的结果不同。 幻读&#xff1a; 一个事务在前后两次查询同一个范围的时候&#xff0c;后一次查询看到了前一次查询没有看到的行。 二…

MYSQL--一条SQL执行的流程,分析MYSQL的架构

文章目录 第一步建立连接第二部解析 SQL第三步执行 sql预处理优化阶段执行阶段索引下推 执行一条select 语句中间会发生什么&#xff1f; 这个是对 mysql 架构的深入理解。 select * from product where id 1;对于mysql的架构分层: mysql 架构分成了 Server 层和存储引擎层&a…

ReentrantReadWriteLock源码分析

文章目录 概述一、状态位设计二、读锁三、锁降级机制四、写锁总结 概述 ReentrantReadWriteLock&#xff08;读写锁&#xff09;是对于ReentranLock&#xff08;可重入锁&#xff09;的一种改进&#xff0c;在可重入锁的基础上&#xff0c;进行了读写分离。适用于读多写少的场景…

51单片机开发:温度传感器

温度传感器DS18B20&#xff1a; 初始化时序图如下图所示&#xff1a; u8 ds18b20_init(void){ds18b20_reset();return ds18b20_check(); }void ds18b20_reset(void){DS18B20_PORT 0;delay_10us(75);DS18B20_PORT 1;delay_10us(2); }u8 ds18b20_check(void){u8 time_temp0;wh…

vue2项目(一)

项目介绍 电商前台项目 技术架构&#xff1a;vuewebpackvuexvue-routeraxiosless.. 封装通用组件登录注册token购物车支付项目性能优化 一、项目初始化 使用vue create projrct_vue2在命令行窗口创建项目 1.1、脚手架目录介绍 ├── node_modules:放置项目的依赖 ├──…

labelme_json_to_dataset ValueError: path is on mount ‘D:‘,start on C

这是你的labelme运行时label照片的盘和保存目的地址的盘不同都值得报错 labelme_json_to_dataset ValueError: path is on mount D:,start on C 只需要放一个盘但可以不放一个目录

物联网 STM32【源代码形式-使用以太网】连接OneNet IOT从云产品开发到底层MQTT实现,APP控制 【保姆级零基础搭建】

物联网&#xff08;IoT&#xff09;‌是指通过各种信息传感器、射频识别技术、全球定位系统、红外感应器等装置与技术&#xff0c;实时采集并连接任何需要监控、连接、互动的物体或过程&#xff0c;实现对物品和过程的智能化感知、识别和管理。物联网的核心功能包括数据采集与监…

无心剑七绝《深度求索》

七绝深度求索 深研妙理定乾坤 度世玄机启智门 求路千难兼万险 索萦华夏自为尊 2025年2月1日 平水韵十三元平韵 无心剑七绝《深度求索》以平水韵十三元平韵写成&#xff0c;意境深远&#xff0c;气势磅礴。诗中“深研妙理定乾坤”开篇点题&#xff0c;展现出对深奥道理的钻研与探…

Hot100之普通数组

53最大子数组和 题目 思路解析 我们用一个dp数组来收集我们从左往右&#xff0c;加起来的最大的和 也就是我们的节点不是负数&#xff0c;那我们直接收集就好了 如果是负数&#xff0c;我们就用Max&#xff08;&#xff09;比较是这个节点大还是当前节点大&#xff08;这个情…

如何利用天赋实现最大化的价值输出-补

原文&#xff1a; https://blog.csdn.net/ZhangRelay/article/details/145408621 ​​​​​​如何利用天赋实现最大化的价值输出-CSDN博客 如何利用天赋实现最大化的价值输出-CSDN博客 引用视频差异 第一段视频目标明确&#xff0c;建议也非常明确。 录制视频的人是主动性…

新能源算力战争:为什么AI大模型需要绿色数据中心?

新能源算力战争:为什么AI大模型需要绿色数据中心? 近年来,人工智能(AI)大模型的爆发式增长正在重塑全球科技产业的格局。以GPT-4、Gemini、Llama等为代表的千亿参数级模型,不仅需要海量数据训练,更依赖庞大的算力支撑。然而,这种算力的背后隐藏着一个日益严峻的挑战——…

Spring Boot 中的事件发布与监听:深入理解 ApplicationEventPublisher(附Demo)

目录 前言1. 基本知识2. Demo3. 实战代码 前言 &#x1f91f; 找工作&#xff0c;来万码优才&#xff1a;&#x1f449; #小程序://万码优才/r6rqmzDaXpYkJZF 基本的Java知识推荐阅读&#xff1a; java框架 零基础从入门到精通的学习路线 附开源项目面经等&#xff08;超全&am…