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.mzph.cn/news/894746.shtml

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

相关文章

Python----Python高级(并发编程:进程Process,多进程,进程间通信,进程同步,进程池)

一、进程Process 拥有自己独立的堆和栈&#xff0c;既不共享堆&#xff0c;也不共享栈&#xff0c;进程由操作系统调度&#xff1b;进程切换需要的资源很最大&#xff0c;效率低。 对于操作系统来说&#xff0c;一个任务就是一个进程&#xff08;Process&#xff09;&#xff…

在Mapbox GL JS中“line-pattern”的使用详解

在Mapbox GL JS中&#xff0c;line-pattern 是一种用于在地图上绘制带有图案的线条的样式属性。通过 line-pattern&#xff0c;你可以使用自定义的图像作为线条的图案&#xff0c;而不是使用纯色或渐变。 1. 基本概念 line-pattern: 该属性允许你指定一个图像作为线条的图案。…

C++ Primer 算术运算符

欢迎阅读我的 【CPrimer】专栏 专栏简介&#xff1a;本专栏主要面向C初学者&#xff0c;解释C的一些基本概念和基础语言特性&#xff0c;涉及C标准库的用法&#xff0c;面向对象特性&#xff0c;泛型特性高级用法。通过使用标准库中定义的抽象设施&#xff0c;使你更加适应高级…

【大数据技术】本机PyCharm远程连接虚拟机Python

本机PyCharm远程连接虚拟机Python 注意:本文需要使用PyCharm专业版。 pycharm-professional-2024.1.4VMware Workstation Pro 16CentOS-Stream-10-latest-x86_64-dvd1.iso写在前面 本文主要介绍如何使用本地PyCharm远程连接虚拟机,运行Python脚本,提高编程效率。 注意: …

堆(Heap)的原理与C++实现

1. 什么是堆&#xff1f; 堆&#xff08;Heap&#xff09;是一种特殊的树形数据结构&#xff0c;通常用于实现优先队列。堆可以分为两种类型&#xff1a; 最大堆&#xff08;Max Heap&#xff09;&#xff1a;每个节点的值都大于或等于其子节点的值。最小堆&#xff08;Min H…

移除元素-双指针(下标)

题目 给你一个数组 nums 和一个值 val&#xff0c;你需要 原地 移除所有数值等于 val 的元素。元素的顺序可能发生改变。然后返回 nums 中与 val 不同的元素的数量。 假设 nums 中不等于 val 的元素数量为 k&#xff0c;要通过此题&#xff0c;您需要执行以下操作&#xff1a…

log4j2日志配置文件

log4j2配置文件每个项目都会用到,记录一个比较好用的配置文件,方便以后使用时调取,日志输出级别为debug,也可以修改 <?xml version"1.0" encoding"UTF-8"?> <Configuration monitorInterval"180" packages""><prope…

高等代数笔记—映射与线性空间

映射 映射&#xff1a; σ : M → M ′ \sigma: M \to M σ:M→M′ σ ( a ) a ′ , a ∈ M , a ′ ∈ M ′ \sigma(a)a, a\in M, a \in M σ(a)a′,a∈M,a′∈M′ a ′ a a′是 a a a在 σ \sigma σ下的像&#xff0c; a a a是 a ′ a a′在 σ \sigma σ下的原像 σ : …

提示词实践总结

目录 一、要求创建SqlServer表&#xff08;ChatGpt&#xff09; 二、要求生成多层架构代码&#xff08;Cursor&#xff09; 三、要求修改方法返回值类型&#xff08;Cursor&#xff09; 四、要求修改方法入参&#xff08;Cursor&#xff09; 五、复杂的多表关联生成&#…

java进阶文章链接

java 泛型&#xff1a;java 泛型详解-绝对是对泛型方法讲解最详细的&#xff0c;没有之一 Java 泛型&#xff0c;你了解类型擦除吗&#xff1f; java 注解&#xff1a;深入理解Java注解类型 秒懂&#xff0c;Java 注解 &#xff08;Annotation&#xff09;你可以这样学 jav…

MyBatis-Plus笔记-快速入门

大家在日常开发中应该能发现&#xff0c;单表的CRUD功能代码重复度很高&#xff0c;也没有什么难度。而这部分代码量往往比较大&#xff0c;开发起来比较费时。 因此&#xff0c;目前企业中都会使用一些组件来简化或省略单表的CRUD开发工作。目前在国内使用较多的一个组件就是…

Maven jar 包下载失败问题处理

Maven jar 包下载失败问题处理 1.配置好国内的Maven源2.重新下载3. 其他问题 1.配置好国内的Maven源 打开⾃⼰的 Idea 检测 Maven 的配置是否正确&#xff0c;正确的配置如下图所示&#xff1a; 检查项⼀共有两个&#xff1a; 确认右边的两个勾已经选中&#xff0c;如果没有请…

Spring 核心技术解析【纯干货版】- IX:Spring 数据访问模块 Spring-Jdbc 模块精讲

在现代企业级应用中&#xff0c;数据访问层的稳定性和高效性至关重要。为了简化和优化数据库操作&#xff0c;Spring Framework 提供了 Spring-JDBC 模块&#xff0c;旨在通过高度封装的 JDBC 操作&#xff0c;简化开发者的编码负担&#xff0c;减少冗余代码&#xff0c;同时提…

探秘AI的两大核心:决策式AI与生成式AI‌

目录 一、引言 二、从定义上来看 1. 决策式AI&#xff08;Discriminative AI&#xff09; 2. 生成式AI&#xff08;Generative AI&#xff09; 三、从技术原理上来看 1. 决策式AI&#xff08;Discriminative AI&#xff09; 2. 生成式AI&#xff08;Generative AI&#…

2.5学习

misc buuctf-假如给我三天光明 下载附件后得到了一个压缩包和一个图片&#xff0c;压缩包为加密压缩包&#xff0c;需要解出密码&#xff0c;然后注意到这个图片并非简单的一个封面&#xff0c;在下方还有诸多点&#xff0c;有黑有灰。经过搜索&#xff0c;发现这是盲文通过与…

sed变量中特殊字符/处理方式

个人博客地址&#xff1a;sed变量中特殊字符/处理方式 | 一张假钞的真实世界 如果变量值中包含斜杠&#xff08;/&#xff09;特殊字符&#xff0c;在使用sed命令的做行内字符串替换时可以使用井号&#xff08;#&#xff09;做为sed语法分隔符&#xff0c;如下&#xff1a; G…

java进阶1——JVM

java进阶——JVM 1、JVM概述 作用 Java 虚拟机就是二进制字节码的运行环境&#xff0c;负责装载字节码到其内部&#xff0c;解释/编译为对 应平台上的机器码指令行&#xff0c;每一条 java 指令&#xff0c;java 虚拟机中都有详细定义&#xff0c;如怎么取操 作数&#xff0c…

搭建集成开发环境PyCharm

1.下载安装Python&#xff08;建议下载并安装3.9.x&#xff09; https://www.python.org/downloads/windows/ 要注意勾选“Add Python 3.9 to PATH”复选框&#xff0c;表示将Python的路径增加到环境变量中 2.安装集成开发环境Pycharm http://www.jetbrains.com/pycharm/…

vue2-v-if和v-for的优先级

vue2-v-if和v-for的优先级 1.v-if和v-for的作用 v-if是条件渲染&#xff0c;只有条件表达式true的情况下&#xff0c;才会渲染v-for是基于一个数组来渲染一个列表&#xff0c;在v-for的时候&#xff0c;保证给每个元素添加独一无二的key值&#xff0c;便于diff算法进行优化 …

通过C/C++编程语言实现“数据结构”课程中的链表

引言 链表(Linked List)是数据结构中最基础且最重要的线性存储结构之一。与数组的连续内存分配不同,链表通过指针将分散的内存块串联起来,具有动态扩展和高效插入/删除的特性。本文将以C/C++语言为例,从底层原理到代码实现,手把手教你构建完整的链表结构,并深入探讨其应…