百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术文章 > 正文

VC|创建、动态显式、静态隐式调用或链接动态链接库DLL

cac55 2024-12-16 11:16 41 浏览 0 评论

编写应用程序通常都不是从0开始的,或多或少都会用到一些库,不管是函数库还是类库。

动态链接库DLL(dynamic linked library)通常都不能直接运行,也不能接收消息。它们是一些独立的文件,其中包含能被可执行程序或其它DLL调用来完成某项工作的函数。只有在其它模块调用动态链接库中的函数时,它才发挥作用。

动态链接可以在编译时加载,也可以在运行时加载,加载时DLL被映射到进程的地址空间中。

1 创建动态链接库dll

1.1 建立一个简单的win32 dll工程mydll

1.2 添加mydll.h文件作为使用dll的接口(隐式链接时使用),并添加导出的函数fnTest()的声明:

int fnTest(void);

1.3 在mydll.cpp中添加需要导出的函数fnTest()

// mydll.cpp : Defines the entry point for the DLL application.
//

#include "stdafx.h"

BOOL APIENTRY DllMain( HANDLE hModule, 
                       DWORD  ul_reason_for_call, 
                       LPVOID lpReserved
					 )
{
    return TRUE;
}
int fnTest(void)
{
    return 100;
}

1.4 在工程中添加如下内容的mydll.def文件

; mydll.def : Declares the module parameters for the DLL.

LIBRARY      "mydll"
DESCRIPTION  'mydll Windows Dynamic Link Library'

EXPORTS
	fnTest

从DLL中导出函数,可以使用模块定义文件(DEF),由一个或多个用于描述DLL属性的语句组成的文本文件,常用的语句如下,

LIBRARY语句:设置DLL的内部名。

DESCRIPTION语句:设置DLL的描述。

EXPORTS语句:设置被导出函数的名称。

SECTIONS语句:设置段属性。

以上代码运行会生成一个mydll.dll、mydll.lib文件。

?.h头文件,包含dll中说明输出的类或符号原型或数据结构的.h文件。应用程序静态隐式调用dll时,需要将该文件包含入应用程序的源文件中。

?.LIB文件,是dll在编译、链接成功之后生成的文件,作用是当其他应用程序静态隐式调用dll时,需要将该文件引入应用程序。

? dll文件,包含函数定义的文件,开发成功后的应用程序在发布时,只需要有.exe文件和.dll文件,并不需要.lib文件和.h头文件。

2 动态显式调用或链接dll

动态调用方式是由编程者用 API 函数加载和卸载 DLL 来达到调用 DLL 的目的,使用上较复杂,但能更加有效地使用内存,是编制大型应用程序时的重要方式。

显式的调用是指在应用程序中用 LoadLibrary 或 MFC 提供的 AfxLoadLibrary 显式的将自己所做的动态连接库调进来,动态连接库的文件名即是上面两个函数的参数,再用 GetProcAddress() 获取想要引入的函数指针。自此,你就可以象使用如同本应用程序自定义的函数一样来调用此引入函数了。在应用程序退出之前,应该用 FreeLibrary 或 MFC 提供的 AfxFreeLibrary 释放动态连接库。

因为DLL 有占用内存小、好编辑等的特点,有很多电脑病毒都是DLL格式文件。

显式链接是指应用程序在运行时通过函数调用来显示加载DLL,并通过函数指针来调用 DLL的导出函数。

显式链接首先调用SDK的LoadLibrary函数加载DLL,然后调用GetProcAddress函数获取导出函数的地址,最后使用完毕时调用FreeLibrary函数释放DLL。一些函数定义如下:

HMODOLE LoadLibrary(LPCTSTR IpFileName);

FARPROC GetProcAddress (HMODULE hModule, LPCSTR IpProcName);

BOOL FreeLibrary(HMODULE hModule);

2.1 创建一个对话框工程Usedll;

2.2 将mydll.dll复制到当前路径;

2.3 添加按钮和消息响应函数OnExUsedll():

void CUsedllDlg::OnExUsedll() 
{
	//加载DLL
	HINSTANCE hModule = LoadLibrary(_T("mydll.dll")); // dll要复制到相同路径下
	if (hModule == NULL)
	{
		AfxMessageBox(_T("mydll.dll加载失败\n"));
		return;
	}

	typedef int (_cdecl *FUNTEST)(void);
	FUNTEST pfnTest;

	//获得导出函数的地址
	pfnTest = (FUNTEST)GetProcAddress(hModule, "fnTest");

	//调用导出函数
	if (pfnTest != NULL)
	{
		int nValue = (*pfnTest)();
		CString strMessage = _T("");
		strMessage.Format(_T("%d"), nValue);
		AfxMessageBox(strMessage);
	}
	else
	{
		int n = GetLastError();
		TRACE(_T("LastError:%d\n"), n);	
	}

	//释放DLL
	FreeLibrary(hModule);
}

3 静态隐式调用或链接dll

程序员在建立一个 DLL 文件时,链接程序会自动生成一个与之对应的 LIB 导入文件。该文件包含了每一个 DLL 导出函数的符号名和可选的标识号,但是并不含有实际的代码。LIB 文件作为 DLL 的替代文件被编译到应用程序项目中。

隐式调用需要把产生动态连接库时产生的 .LIB 文件加入到应用程序的工程中,想使用 DLL 中的函数时,只须说明一下。隐式调用不需要调用 LoadLibrary() 和 FreeLibrary()。

当程序员通过静态链接方式编译生成应用程序时,应用程序中的调用函数与 LIB 文件中导出符号相匹配,这些符号或标识号进入到生成的 EXE 文件中。LIB 文件中也包含了对应的 DL L文件名(但不是完全的路径名),链接程序将其存储在 EXE 文件内部。

隐式链接需要应用程序在编译时链接包含导出函数的头文件、.dll、.lib文件,在运行时加载DLL,直接调用DLL的导出函数。

3.1 将mydll.h、mydll.lib复制到上面工程的当前路径(也需要mydll.dll);

3.2 在UsedllDlg.cpp中添加以下内容:

// UsedllDlg.cpp : implementation file

//DLL导出函数的头文件
#include "mydll.h"

//DLL的导入库lib文件,编译是通过静态链接库(lib)去找接口的
#pragma comment(lib, "mydll.lib")

3.3 添加按钮和消息响应函数OnImUsedll()

void CUsedllDlg::OnImUsedll() 
{
	// 直接调用DLL的导出函数
	int nValue = fnTest();

	CString strMessage = _T("");
	strMessage.Format(_T("%d"), nValue);
	AfxMessageBox(strMessage);
}

lib是编译时需要的,dll是运行时需要的。如果要完成源代码的编译,有lib就够了。如果也是动态连接的程序运行起来,有dll就够了。

如果有dll文件,那么对应的lib文件一般是一些索引信息,具体的实现在dll文件中。

如果只有lib文件,那么这个lib文件是静态编译出来的,索引和实现都在其中。

静态编译的lib文件有好处,给用户安装时就不需要再挂动态库了。但也有缺点,就是导致应用程序比较大,而且失去了动态库的灵活性,在版本升级时,同时要发布新的应用程序才行。

-End-

相关推荐

Linux服务器被黑客入侵后各排查项及排除步骤

Linux入侵排查0x00前言当企业发生黑客入侵、系统崩溃或其它影响业务正常运行的安全事件时,急需第一时间进行处理,使企业的网络信息系统在最短时间内恢复正常工作,进一步查找入侵来源,还原入侵事故...

[常用工具] Python视频处理库VidGear使用指北

VidGear是一个高性能的Python视频处理库,它在预载多个专业视频图像处理库的基础上,如OpenCV、FFmpeg、ZeroMQ、picamera、starlette、yt_dlp、pyscre...

微信公众号自动回复及多客服功能实现

目录前期准备1、微信公众平台基本设置2、开发所需参数功能步骤1、填写服务器配置2、验证服务器地址的有效性3、依据接口文档实现业务逻辑具体实现1、微信接入2、自定义回复及多客服接入默认微信公众平台对公众...

电脑病毒怎么彻底清理?这3个方法可以解决!

案例:电脑中毒无法正常使用怎么办?怎么清理电脑病毒?如何彻底清除病毒?有没有小伙伴知道解决的方法?在使用电脑的过程中,我们经常会遇到电脑中病毒的情况,它们能够通过各种渠道感染你的计算机系统,给你带来许...

人在低谷落难的时候,一定要记住的4句话

凌晨三点在便利店啃面包时,我看见邻座大哥对着手机里的存款余额发呆,手指在屏幕上划了又划——原来成年人的崩溃,真的会藏在每个看似普通的深夜里。如果你也正在经历「人生断电期」,这10句从谷底爬起来的人总结...

Linux环境Docker容器安装与使用(六)——安装Hadoop大数据集群

简介:Hadoop是一种分析和处理大数据的软件平台,是Appach开源软件的一个架构,在大量计算机组成的集群当中实现了对于海量的数据进行的分布式计算。Hadoop框架最核心的设计就是HDFS和MapR...

(2023年最新)50个超实用电脑实用快捷键,提高操作效率10倍!

我们现在大多数工作都需要使用电脑,掌握简单的电脑知识,可以更好的提高操作效率,熟能生巧是没错,但还有一个方法就是使用快捷键。办公室文员必备技能知识;基本要求:打字快,会office办公软件(word文...

升级WIN10毛病多?解决这些问题,轻松应对!

1、win10网络不稳定①打开设置,进入网络和INTERNET。②在“WLAN页面”选择“管理Wi-Fi设置”。③在此页面上有个管理已知网络,里面记录着之前电脑连接过的无线网络连接,点击“连接名称”,...

史上最贱最贱的电脑病毒!(最致命的电脑病毒)

看了标题,有很多人是充满好奇心进来的,想看看有多贱!我可以郑重的告诉你,贱到你想掐死黑客!下面我给你介绍一下这个病毒是怎么个贱法!因为我亲身体验了一把!前几天我不知道怎么回事,我电脑莫名其妙多了几...

五千字长文全平台笔记软件obsidian同步攻略&图床使用教程

全平台笔记软件obsidianobsidian(黑曜石)是一个全平台的笔记软件,基础笔记功能免费,如果使用官方的同步功能好像是收费(我也不确定,因为我甚至没登陆过obsidian的账号)。可以使用ma...

工业自动化2.0演进:具有自我意识的运动控制

工业自动化领域的下一个发展方向要求机器能够独立调整其性能参数,以完成工厂操作人员分配的任务,或根据生产力增强的人工智能(AI)算法的输入,对机器自身重新配置以优化其行为。具有自我意识的机器的价值在于,...

零信任的时代到来!VPN将逐渐被取代

转自NETWORKWORLD,作者NealWeinberg,蓝色摩卡译,合作站点转载请注明原文译者和出处为超级盾!传统的VPN正在被一种更智能、更安全的网络安全方法所取代,这种方法将每个人都视为不受...

电脑键盘指法+常用快捷键文字及图片详解

图1:20190820(整理)(较全面的在后面)Ctrl+N:新建文档F4:重复上述操作Esc:取消当前操作HOME:光标跳转行首END:光标跳转到行尾WIN+L:锁定桌面WIN+E:开启磁...

VPN正在消亡,零信任万岁

转自NETWORKWORLD,作者NealWeinberg,蓝色摩卡译,合作站点转载请注明原文译者和出处为超级盾!传统的VPN正在被一种更智能、更安全的网络安全方法所取代,这种方法将每个人都视为不受...

Windows自带的「黑科技」工具,能让你少装10个软件!

电脑装了一堆软件,桌面却还是乱糟糟?其实Windows系统里藏着一堆“神器”,无需第三方工具就能搞定截图、录屏、OCR文字提取、系统加速……这7个冷门但逆天的内置工具,专治“软件成瘾症”,看完立马卸载...

取消回复欢迎 发表评论: