跨模块边界分配和释放内存

news2024/12/28 20:09:36

我想,有一条编程铁律已经深深的刻入到你的头脑中了:使用成套的函数来分配和释放内存,例如,如果使用 LocalAlloc 分配内存,则应该使用 LocalFree,类似的例子还有:GlobalAlloc 对应 GlobalFree,new [] 对应 delete []。

但这条规则还有更深层次的潜规则。

如果你提供了一个函数分配了一段内存并返回了数据,则调用者必须知道如何释放该内存。
实现这个目标有很多不同的方法,其中一个是在函数的文档中显式的声明内存应该如何释放。例如,FormatMessage 这个函数在文档中清楚的表达了调用者应该使用 LocalFree 来释放已分配的内存(如果你传递了
FORMAT_MESSAGE_ALLOCATE_BUFFER 标志的话)。

所有的 BSTR 变量必须使用 SysFreeString 来释放。跨 COM 接口边界返回的所有内存必须使用 COM 内存分配器分配和释放。

但是,请注意,如果你决定使用 C 运行时(例如使用 free)释放内存块,或者通过 C++ 运行时的 delete 或 delete[] 释放 ,则会遇到一个新问题:哪个运行时?
如果选择与静态运行时库链接,则模块具有自己的 C/C++ 运行时专用副本。当模块调用 new 或 malloc 时,只能通过模块调用 delete 或 free 来释放内存。如果另一个模块调用 delete 或 free,这将使用与你的模块不同的其他模块的 C/C++ 运行时。

事实上,即使你选择与 C/C++ 运行时库的 DLL 版本链接,你仍然必须同意使用哪个版本的 C/C++ 运行时。如果 DLL 使用 MSVCRT20.DLL 来分配内存,则任何想要释放该内存的人也必须使用 MSVCRT20.DLL。

如果你仔细观察,你可能会发现一个迫在眉睫的问题。如果你控制所有客户端并愿意在每次编译器更改时重新编译所有客户端,则要求所有客户端使用特定版本的 C/C++ 运行时似乎是合理的。

但在现实生活中,人们往往不想冒这种风险。”如果它程序还能跑,就不要管它。” 切换到新的编译器可能会暴露一个微妙的错误,例如,忘记将变量声明为易失性(volatile)变量,或者无意中依赖于具有特定生存期的临时变量。

在实践中,你可能希望仅将程序的一部分转换为新的编译器,而不理会旧模块。(例如,你可能希望利用新的语言功能,如模板,这些功能仅在新编译器中可用。) 但是,如果这样做,则无法释放旧 DLL 分配的内存,因为该 DLL 希望你使用 MSVCRT20.DLL,而新编译器使用 MSVCR71.DLL。

解决这个问题的方法是:提前做好规划。

一种方法是:使用固定的外部分配器,例如 LocalAlloc 或 CoTaskMemAlloc。这些都是通用的内存分配器,不依赖于所使用的编译器版本。

另一种方法:将内存分配器封装在管理分配的导出函数中。这是 NetApi 系列函数使用的机制。例如,NetGroupEnum 函数分配内存并通过 bufptr 参数返回内存。调用方使用完内存后,它会使用 NetApiBufferFree 函数释放内存。以这种方式,内存分配方法与调用方隔离。在内部,NetApi 函数可能使用 LocalAlloc 或 HeapAllocate 或 HeapAllocate 函数,甚至可能是 new 或者 free。这不重要; 只要 NetApiBufferFree 使用最初用于分配内存的 NetGroupEnum 所用的相同分配器释放内存。

虽然我个人更喜欢使用固定的外部分配器,但许多人发现使用包装器技术更方便。
这样,他们就可以在整个模块中使用自己喜欢的分配器。无论哪种方式都可以正常工作。关键是,当内存离开 DLL 时,你提供内存的代码必须知道如何释放它,即使它使用的编译器与用于生成 DLL 的编译器不同。

总结

作为一名 C++ 程序员,你不能像其他带 GC 语言的程序员那样粗心大意。
对内存的管理,需要你的头脑十分冷静和清醒,一旦出现内存方面的问题,你不得不花费巨量的时间来追踪。
但这个解决问题的过程也是十分有乐趣的,这是对 C++ 程序员的馈赠。

最后

Raymond Chen的《The Old New Thing》是我非常喜欢的博客之一,里面有很多关于Windows的小知识,对于广大Windows平台开发者来说,确实十分有帮助。
本文来自:《Allocating and freeing memory across module boundaries》

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

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

相关文章

手把手教你使用Vue2.0实现动态路由展示

文章目录 VUE2相关组件版本原始菜单数据数据库数据树形数据 前端项目配置静态路由配置路由守卫左侧路由回显代码 使用Vue2实现动态路由展示 思路: 后端返回树形数据根据数据渲染左侧菜单根据数据组装路由数据 注意:本文主要目的讲解是一种思路&#xff0…

47GB水经微图从入门到精通视频教程

本视频教程共47GB,为了方便大家观看,同时录制了横版视频教程和竖版视频教程。 本视频教程的内容主要包括快速入门、地图标注、影像下载、高程下载和矢量下载几部分。 本文将列出所有视频教程所有内容。 快速入门(23.1GB) 如何…

串口实用解说

我们学习单片机,首先接触的可能是点灯(GPIO),再次就是串口(UART)。 串口是常用的一种通信接口,也是学嵌入式必备掌握的一项知识,但我发现有很多小伙伴只知道用串口输出或者打印一些数…

《windows核心编程》第4章 进程

一、进程基本概念 1、进程:一个进程就是一个正在运行的程序,一个程序可以产生多个进程。进程包含下面两个东西 ● 进程内核对象:一个内核对象被系统用来管理某个进程,内核对象就是代表这个进程。这个内核对象中,还包…

CTS分析思路

目录 原理简介: Cts测试原理: CTS报告与日志目录 CTS报告目录如下​编辑 log查看 举例 原理简介: Cts环境搭建和测试方法,大家可以自行查询网上资料。 Cts测试原理: 输入命令后,会安装一系列的测试…

短视频账号矩阵系统/剪辑/矩阵/无人直播/文案引流爆款

一、 短视频账号矩阵源码开发包含哪几方面? 1. 界面设计:需要根据用户需求,设计出优美简洁的UI界面,使用户可以方便快捷地管理自己的短视频账号。 2. 数据存储:需要将用户的账号信息、数据统计等信息存储在数据库中&a…

pytorch dropout 置零 + 补偿性放缩

一句话概括:(训练过程中)Dropout 操作 随机置零 非置零元素进行后补偿性放缩。以保证dropout前后数据scale不变。 详细解释(来自chatgpt): 在 PyTorch 中,dropout 的操作不仅仅是将某些元素置零。为了确保期望输出在训练和测试…

WoShop跨境电商商城源码(多语言多货币多商户进出口电商平台)

一、跨境电商商城系统源码包括以下几个部分 前端框架:uni-app,vue 后端框架:ThinkPHP5.wokerman 支付系统:PayPal、USDT等主流支付平台 语言包:跨境电商支持15种语言,后续会增加 前端:包含APP端、小程序端、…

无需数据搬迁,10倍性能提升!携程的统一分析之旅

作者:携程技术中心大数据总监 许鹏 携程自 2022 年起引入了 StarRocks,目前已经成为了集团内部的主要技术栈,应用到酒店、机票、商旅、度假、市场、火车票等多个关键业务线。目前,携程内部已经拥有超过 10 个 StarRocks 集群&…

c语言函数宏的几种封装方式

c语言函数宏的几种封装方式 在c语言开发中,除了使用函数封装代码之外,也经常使用宏来封装一些重要或简洁的代码。 宏在c开发有三种:预定义宏,不带参宏,和带参数宏,通常,带参数宏也叫函数宏&am…

HR应用人才测评来提升人才价值

对于企业而言,产出价值是最重要的,但企业拥有人才,才能产出更多的价值,做HR这么久发现很多企业,都欠缺人才管理的测评技术,导致很多人才被埋没或者浪费,这也说明一个很大的问题,一定…

Github 自动化部署到GitHub Pages

1.准备工作 新建仓库 新建项目 配置 vite.config.ts base: ./,部署应用包时的基本URL,例:vue-cli 5.x 配置 publicPath 推送到远程仓库 2.配置 GitHub Token 点击 Settings -> Actions -> General 找到 Workflow permissions,选中第…

sqlserver涉及到三种排序后生成的数字

with temp as (select 1 标识,2023-01-01 日期,a 项目union all select 1,2023-01-01,a union all select 2,2023-01-01,a union all select 2,2023-01-01,b union all select 3,2023-01-01,a union all select 3,2023-01-01,b union all select 3,2023-01-01,c union all …

耳朵小戴什么耳机合适,2023年适合小耳道的蓝牙无线耳机分享

你们是否曾为了追求音乐的同时,担心自己的听力健康呢?尤其是耳朵小的群体,佩戴入耳式时间一长,就会感觉耳道存在一定的疼痛感,不过别担心,现在有了一种完美的解决方案——骨传导耳机!这种炫酷的…

解决Windows Server 2012 由于没有远程桌面授权服务器可以提供需求可证

刚开始提示 之后就登录不了 (如下图提示) 由于windows server 2012 R2 安装了 远程桌面角色,但是这个角色是120天免费的,需要购买授权的。解决方法是取消/删除这个角色,就可以恢复正常的远程 一直下一步 远程桌面服…

2023年阿里云双11活动云服务器可选实例规格、配置及活动价格分享

阿里云服务器2023年双11活动价格是多少?轻量应用服务器2核2G3M带宽轻量服务器87元1年、2核4G4M带宽165元1年;云服务器经济型e实例2核2G3M配置99元1年;计算型c7实例2核4G1M配置864.79元1年;通用型g7实例2核8G1M配置1089.91元1年&am…

Linux C语言进阶-D9字符指针与字符串

初始化字符指针:把内存中字符串的首地址赋予指针,并不是把该字符串复制到指针中。 char str[] "Hello World"; char *p str; 在C编程中,当一个字符指针指向一个字符串常量时,不能修改指针指向的对象的值 char * p &…

【PC】第26赛季第2轮更新公告

正式服维护日期 ※ 下列时间可能会根据维护情况而发生变化。 11月8日上午8:00 – 下午4:30 地图轮换 ※ 地图轮换将于北京时间每周三上午10点进行。 ※ 在随机选择地图的区域中,各地图将按大型地图25%、小型地图12.5%的概率随机匹配。 测试服 普通比赛&#xf…

Capybara库如何批量下载新浪图片

按照要求写一个使用Capybara库的下载程序。该程序使用Ruby下载新浪新闻的图片,并使用爬虫IP服务器duoip的8000端口进行下载。 require capybara require mechanize# 创建一个爬虫IP服务器实例 proxy Mechanize.new爬虫IP主机: duoip,爬虫IP端口: 8000# 访问新浪新闻…

黑马最新「SpringBoot3+Vue3」全套教程上线,练手很香

转眼2023年仅剩2个月,大家的“卷”也进入了白热化阶段,毕竟10月随份子的钱还没还完,双11又付了一大笔尾款,还要准备回家过年的钱……为了更卷(赚更多钱),又又又有程序员来找播妞要新教程了。 “…