Windows核心编程_动态链接库_DLL基础_DLL与进程的地址空间

本文主要是介绍Windows核心编程_动态链接库_DLL基础_DLL与进程的地址空间,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

通常创建一个DLL要比建立一个应用程序容易,因为DLL包含一组可以被任何应用所使用的自治(autonomous)函数。在DLL中通常没有支持处理消息循环或创建窗口的代码。一个DLL只是一组源代码模块的集合。其中每个模块包含应用程序(可执行文件)或另一个DLL将要调用的一组函数。当所有的源代码文件编译后,就像应用程的可执行文件那样被链接程序所链接。然而,对于一个DLL,必须要为链接程序设定DLL开关选项。该选项使得链接程序向生成的DLL文件映像发出略有差异的信息,这样,操作系统装载程序就可以将该文件映像识别为一个DLL,而不是应用程序。
在应用程序(或另一个DLL)可以调用DLL中的函数之前,DLL的文件映射必须映射到调用进程的地址空间。可以采用两种方法来完成这项操作:隐式的装载时链接或显示的运行时链接,隐式链接在本章后续部分加以讨论,显示链接将在第20章中进行讨论。
一旦DLL的文件映射到调用进程的地址空间,则DLL中的函数对于运行在该进程内的所有线程都是可用的。实际上DLL几乎失去了它作为DLL的所有特征标志。对于进程中的线程而言,DLL的代码和数据开起来就像是恰好存在于进程地址空间中的额外代码和数据。当一个线程调用DLL函数时,该函数查看线程的栈以检索被传递的参数,并且将线程的栈空间用于它所需要的任意局部变量。另外,被DLL中的函数代码所创建的任何对象都属于调用它的线程或进程——一个DLL并不拥有任何元素。
例如,如果DLL中的一个函数所调用VirtualAlloc,则将从调用线程的进程地址空间中保留该地址空间的区域。如果后来从进程的地址空间中取消DLL的映射,则该地址空间的区域仍然会保留,因为系统并没有跟踪记录这样一个事实:DLL中的函数保留着该区域。被保留的地址空间区域是属于调用进程的,只有当一个线程调用VirtualFree函数或该进程终止时,该保留区域才会释放。
一个可执行文件中的全局变量和静态变量不能被该可执行文件的多个运行实例所共享。在Windows 98中,是通过当可执行文件映射到进程的地址空间时,为该文件的全局变量和静态变量分配存储空间的机制来保证这一点的;而在Windows 2000中,则是通过第13章中讨论的写时复制(copy-on-write)机制来加以保证的。一个DLL中的全局变量和静态变量是以相同的方式处理的。当一个进程把一个DLL的映射文件映射到其地址空间时,系统将同时创建全局数据变量和静态数据变量的实例。
注意:需要引起重视的是,一个单独的地址空间是由一个可执行模块和若干个DLL模块组成的。其中的一些模块可以链接到一个静态版本的C/C++运行时库,还有一些模块可能链接到一个DLL版本的C/C++运行时库,而另外一些模块(如果不是用C/C++编写的)可能根本就不需要C/C++运行时库。许多开发人员都存在一个错误的认识,因为他们忘记了若干个C/C++运行时库可能会存在于一个单一的地址空间中。研究下面的代码:

PVOID DLLFunc()
{// Allocate bloc from DLL's C/C++ run-time heapreturn malloc(100);
}VOID EXEFunc()
{PVOID pv = DLLFunc();// Access the storage pointed to by pv...// Assumes that pv is in EXE's C/C++ run-time heapfree(pv);
}

那么你是怎样考虑的呢?上述代码能够正确运行吗?DLL中的函数所分配的块是否被EXE中的函数所释放?答案是:可能会是这样!因为上述代码并没有提供足够的信息。如果EXE和DLL这两者都链接到DLL版本的C/C++运行时库,则对free调用将会失败。我曾多次见过开发人员编写出类似的代码,结果是毁了代码。
有一个简单的方法可以解决这个问题。当一个模块提供了分配内存的函数时,该模块必须同时提供一个释放内存的函数。重写上述代码如下:

PVOID DLLFunc()
{// Allocate bloc from DLL's C/C++ run-time heapreturn malloc(100);
}VOID DLLFreeFunc(PVOID pv)
{// Free bloc form DLL's C/C++ run-time heapreturn free(pv);
}VOID EXEFunc()
{PVOID pv = DLLFunc();// Access the storage pointed to by pv...// Assumes that pv is in EXE's C/C++ run-time heapfree(pv);
}

经过重写后的代码是正确的,它总是可以正确地执行。当编写一个模块时,不要忘记其它模块中的那些函数,它们甚至可能不是用C/C++编写的,因此可能不会使用malloc和free函数进行内存分配。注意不要在代码中设定这些假设条件。顺便提及,当调用malloc和free函数时,该结论对于C++的new和delete操作符也是相同的。

这篇关于Windows核心编程_动态链接库_DLL基础_DLL与进程的地址空间的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1036067

相关文章

Windows下C++使用SQLitede的操作过程

《Windows下C++使用SQLitede的操作过程》本文介绍了Windows下C++使用SQLite的安装配置、CppSQLite库封装优势、核心功能(如数据库连接、事务管理)、跨平台支持及性能优... 目录Windows下C++使用SQLite1、安装2、代码示例CppSQLite:C++轻松操作SQ

qt5cored.dll报错怎么解决? 电脑qt5cored.dll文件丢失修复技巧

《qt5cored.dll报错怎么解决?电脑qt5cored.dll文件丢失修复技巧》在进行软件安装或运行程序时,有时会遇到由于找不到qt5core.dll,无法继续执行代码,这个问题可能是由于该文... 遇到qt5cored.dll文件错误时,可能会导致基于 Qt 开发的应用程序无法正常运行或启动。这种错

电脑提示xlstat4.dll丢失怎么修复? xlstat4.dll文件丢失处理办法

《电脑提示xlstat4.dll丢失怎么修复?xlstat4.dll文件丢失处理办法》长时间使用电脑,大家多少都会遇到类似dll文件丢失的情况,不过,解决这一问题其实并不复杂,下面我们就来看看xls... 在Windows操作系统中,xlstat4.dll是一个重要的动态链接库文件,通常用于支持各种应用程序

基于Python实现一个Windows Tree命令工具

《基于Python实现一个WindowsTree命令工具》今天想要在Windows平台的CMD命令终端窗口中使用像Linux下的tree命令,打印一下目录结构层级树,然而还真有tree命令,但是发现... 目录引言实现代码使用说明可用选项示例用法功能特点添加到环境变量方法一:创建批处理文件并添加到PATH1

Python包管理工具核心指令uvx举例详细解析

《Python包管理工具核心指令uvx举例详细解析》:本文主要介绍Python包管理工具核心指令uvx的相关资料,uvx是uv工具链中用于临时运行Python命令行工具的高效执行器,依托Rust实... 目录一、uvx 的定位与核心功能二、uvx 的典型应用场景三、uvx 与传统工具对比四、uvx 的技术实

从基础到进阶详解Pandas时间数据处理指南

《从基础到进阶详解Pandas时间数据处理指南》Pandas构建了完整的时间数据处理生态,核心由四个基础类构成,Timestamp,DatetimeIndex,Period和Timedelta,下面我... 目录1. 时间数据类型与基础操作1.1 核心时间对象体系1.2 时间数据生成技巧2. 时间索引与数据

Windows的CMD窗口如何查看并杀死nginx进程

《Windows的CMD窗口如何查看并杀死nginx进程》:本文主要介绍Windows的CMD窗口如何查看并杀死nginx进程问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地... 目录Windows的CMD窗口查看并杀死nginx进程开启nginx查看nginx进程停止nginx服务

安装centos8设置基础软件仓库时出错的解决方案

《安装centos8设置基础软件仓库时出错的解决方案》:本文主要介绍安装centos8设置基础软件仓库时出错的解决方案,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐... 目录安装Centos8设置基础软件仓库时出错版本 8版本 8.2.200android4版本 javas

java中Optional的核心用法和最佳实践

《java中Optional的核心用法和最佳实践》Java8中Optional用于处理可能为null的值,减少空指针异常,:本文主要介绍java中Optional核心用法和最佳实践的相关资料,文中... 目录前言1. 创建 Optional 对象1.1 常规创建方式2. 访问 Optional 中的值2.1

Java进程CPU使用率过高排查步骤详细讲解

《Java进程CPU使用率过高排查步骤详细讲解》:本文主要介绍Java进程CPU使用率过高排查的相关资料,针对Java进程CPU使用率高的问题,我们可以遵循以下步骤进行排查和优化,文中通过代码介绍... 目录前言一、初步定位问题1.1 确认进程状态1.2 确定Java进程ID1.3 快速生成线程堆栈二、分析