”工欲善其事,必先利其器。“—孔子《论语.录灵公》
首页 > 编程 > 实现 malloc() 和 free() — 首先重用旧内存

实现 malloc() 和 free() — 首先重用旧内存

发布于2024-11-08
浏览:337

在本系列关于实现 malloc() 和 free() 的上一篇文章中,我们展示了如何通过释放新块来重用内存块并减少堆。然而,当前的函数引入了一个微妙的问题:它优先考虑重用较新的块,这可能会导致内存消耗随着时间的推移而增加。为什么会发生这种情况?让我们来分解一下。

通过重用最近的块来减少堆

考虑以下场景。首先,我们分配四个内存块:


void *ptr1 = abmalloc(8); 无效 *ptr2 = abmalloc(8); 无效 *ptr3 = abmalloc(8); 无效 *ptr4 = abmalloc(8);
void *ptr1 = abmalloc(8);
void *ptr2 = abmalloc(8);
void *ptr3 = abmalloc(8);
void *ptr4 = abmalloc(8);
内存结构可以这样可视化:

Implementing malloc() and free() — old memory reused first

现在,我们发布第一块和第三块……


abfree(ptr1); abfree(ptr3);
void *ptr1 = abmalloc(8);
void *ptr2 = abmalloc(8);
void *ptr3 = abmalloc(8);
void *ptr4 = abmalloc(8);
...产生以下结构:

Implementing malloc() and free() — old memory reused first

然后我们分配另一个相同大小的块:


void *ptr5 = abmalloc(8);
void *ptr1 = abmalloc(8);
void *ptr2 = abmalloc(8);
void *ptr3 = abmalloc(8);
void *ptr4 = abmalloc(8);
当函数 abmalloc() 开始搜索最近的空闲块时,它会重用顶部的块。如果我们现在释放最后一个块:

Implementing malloc() and free() — old memory reused first

如果我们现在释放最后一个区块……


abfree(ptr4);
void *ptr1 = abmalloc(8);
void *ptr2 = abmalloc(8);
void *ptr3 = abmalloc(8);
void *ptr4 = abmalloc(8);
...我们可以将堆大小减少一个 8 字节块,因为前一个块不再空闲:

Implementing malloc() and free() — old memory reused first

旧区块的再利用

现在,想象一下相同的场景,但进行了一项修改:我们的函数开始从最旧的块开始搜索空闲块。初始结构将是相同的......

Implementing malloc() and free() — old memory reused first

...我们再次释放第一和第三个内存块:

Implementing malloc() and free() — old memory reused first

这次,第一个块将被重用:

Implementing malloc() and free() — old memory reused first

现在,当我们释放最后一个块时,顶部将有两个空闲块,使我们能够将堆减少两个 8 字节块:

Implementing malloc() and free() — old memory reused first

这个例子说明了,通过优先选择较新的块,我们最终会积累旧的未使用的块,浪费内存并导致不必要的堆增长。解决方案是修改搜索策略,优先重用旧块。

实施对旧区块的偏好

为了解决这个问题,我们首先在标头中添加一个指向下一个块的指针。我们还将创建一个指向第一个块的全局指针,这样我们就可以从它开始搜索:


typedef 结构头 { 结构头*上一个,*下一个; size_t 尺寸; 可用布尔值; 标题; 标头*第一个= NULL; 标头*最后= NULL;
void *ptr1 = abmalloc(8);
void *ptr2 = abmalloc(8);
void *ptr3 = abmalloc(8);
void *ptr4 = abmalloc(8);
我们将在两种不同的情况下创建带有标头的内存块,所以让我们进行一个小的重构:我们将这个逻辑提取到一个分配和初始化标头的辅助函数(包括将字段 next 设置为 NULL):


标题 *header_new(标题 *前一个, size_t 大小, bool 可用) { 标头 *header = sbrk(sizeof(标头) 大小); 标题->上一个=上一个; 标题->下一个= NULL; 标题->大小=大小; 标头->可用= false; 返回标头; }
void *ptr1 = abmalloc(8);
void *ptr2 = abmalloc(8);
void *ptr3 = abmalloc(8);
void *ptr4 = abmalloc(8);
通过这个新函数,我们可以简化 abmalloc() 中的逻辑:


void *abmalloc(size_t 大小) { 如果(大小== 0){ 返回空值; } 标题*标题=最后; while (标题!= NULL) { if (标题->可用 && (标题->大小 >= 大小)) { 标头->可用= false; 返回标头1; } 标题=标题->上一页; } 最后 = header_new(最后, 大小, false); 返回最后1个; }
void *ptr1 = abmalloc(8);
void *ptr2 = abmalloc(8);
void *ptr3 = abmalloc(8);
void *ptr4 = abmalloc(8);
现在我们可以访问第一个和最后一个块,并且给定一个块,我们可以找出前一个和下一个块。我们还知道,当指向第一个块的指针为空时,还没有分配任何块。所以在这种情况下,我们将立即分配块,并初始化第一个和最后一个:


void *abmalloc(size_t 大小) { 如果(大小== 0){ 返回空值; } 如果(第一个== NULL){ 第一个 = 最后一个 = header_new(NULL, 大小, false); 返回第一个1; }
void *ptr1 = abmalloc(8);
void *ptr2 = abmalloc(8);
void *ptr3 = abmalloc(8);
void *ptr4 = abmalloc(8);
如果firstit不再为NULL,则已经分配了块,因此我们将开始搜索可重用的块。我们将继续使用变量 header 作为迭代器,但搜索将从最旧的块开始,而不是从最近的块开始:


标头 *标头 = 第一个;
void *ptr1 = abmalloc(8);
void *ptr2 = abmalloc(8);
void *ptr3 = abmalloc(8);
void *ptr4 = abmalloc(8);
在每次迭代时,我们将前进到序列中的下一个块,而不是向后到上一个块:


while (标题!= NULL) { if (标题->可用 && (标题->大小 >= 大小)) { 标头->可用= false; 返回标头1; } 标题=标题->下一个; }
void *ptr1 = abmalloc(8);
void *ptr2 = abmalloc(8);
void *ptr3 = abmalloc(8);
void *ptr4 = abmalloc(8);
逻辑保持不变:如果我们找到足够大小的可用块,则将其返回。否则,如果遍历列表后没有找到可重用的块,则分配一个新块:


最后 = header_new(最后, 大小, false);
void *ptr1 = abmalloc(8);
void *ptr2 = abmalloc(8);
void *ptr3 = abmalloc(8);
void *ptr4 = abmalloc(8);
现在,我们需要调整最后一个块(分配后,倒数第二个)。它指向 NULL,但现在它应该指向新块。为此,我们将前一个块的下一个字段设置为新的最后一个块:


最后一个->上一个->下一个 = 最后一个; 返回最后1个; }
void *ptr1 = abmalloc(8);
void *ptr2 = abmalloc(8);
void *ptr3 = abmalloc(8);
void *ptr4 = abmalloc(8);
abfree() 中的调整

函数 abfree() 基本上保持相同的结构,但现在我们必须处理一些边缘情况。当我们释放堆顶部的块时,一个新块将成为最后一个块,正如我们在此代码段中所做的那样:


最后 = 标题->上一个; brk(标头)
void *ptr1 = abmalloc(8);
void *ptr2 = abmalloc(8);
void *ptr3 = abmalloc(8);
void *ptr4 = abmalloc(8);
这里,指针头引用堆栈上最后一个可用的非空块。我们有两种可能的情况:

    当前块有一个前一个块,它将成为新的最后一个块。在这种情况下,我们应该将该块的next指针设置为NULL。
  1. 当前块没有前一个块(即,它是第一个也是最旧的块)。当它被释放时,堆栈是空的。在这种情况下,我们不会尝试更新不存在的块的字段,而是先将其设置为 NULL,表示不再有分配的块。
我们的实现方式如下:


最后 = 标题->上一个; 如果(最后!= NULL){ 上一个->下一个= NULL; } 别的 { 首先=空; } brk(标头);
void *ptr1 = abmalloc(8);
void *ptr2 = abmalloc(8);
void *ptr3 = abmalloc(8);
void *ptr4 = abmalloc(8);
结论

我们的函数 abmalloc() 和 abfree() 现在看起来像这样:


typedef 结构头 { 结构头*上一个,*下一个; size_t 尺寸; 可用布尔值; 标题; 标头*第一个= NULL; 标头*最后= NULL; 标头 *header_new(标头 *前一个, size_t 大小, bool 可用) { 标头 *header = sbrk(sizeof(标头) 大小); 标题->上一个=上一个; 标题->下一个= NULL; 标题->大小=大小; 标头->可用= false; 返回标头; } 无效* abmalloc(size_t大小){ 如果(大小== 0){ 返回空值; } 如果(第一个== NULL){ 第一个 = 最后一个 = header_new(NULL, 大小, false); 返回第一个1; } 标题*标题=第一个; while (标题!= NULL) { if (标题->可用 && (标题->大小 >= 大小)) { 标头->可用= false; 返回标头1; } 标题=标题->下一个; } 最后 = header_new(最后, 大小, false); 最后一个->上一个->下一个=最后一个; 返回最后1个; } 无效 abfree(无效 *ptr) { 如果(ptr == NULL){ 返回; } 标头 *标头 = (标头*) ptr - 1; 如果(标题==最后){ while ((标题->上一个!= NULL) && 标题->上一个->可用) { 标题=标题->上一页; } 最后=标题->上一个; 如果(最后!= NULL){ 上一个->下一个= NULL; } 别的 { 首先=空; } brk(标头); } 别的 { 标头->可用= true; } }
void *ptr1 = abmalloc(8);
void *ptr2 = abmalloc(8);
void *ptr3 = abmalloc(8);
void *ptr4 = abmalloc(8);
此更改使我们能够节省更多内存。然而,仍有一些问题需要解决。例如,考虑以下场景:我们请求分配 8 字节的内存块,而 abmalloc() 重用了一个 1024 字节的内存块。显然存在浪费。

我们将在下一篇文章中看到如何解决这个问题。

版本声明 本文转载于:https://dev.to/adambrandizzi/implementing-malloc-and-free-old-memory-reused-first-3585如有侵犯,请联系[email protected]删除
最新教程 更多>
  • 当你开始学习编程语言时会发生什么
    当你开始学习编程语言时会发生什么
    在数字时代,学习编程语言不仅是一种优势,而且是一种必要。无论您的目标是提升职业生涯、构建创新应用程序,还是只是更好地了解数字世界,编程技能都是不可或缺的。让我们深入探讨您应该踏上这一变革之旅的原因和方式。 学习编程语言的重要性 职业发展 根据美国劳工统计局的数据,从 ...
    编程 发布于2024-11-08
  • 如何使用匿名结构或联合编译 C 代码?
    如何使用匿名结构或联合编译 C 代码?
    使用匿名结构/联合编译 C 代码出现了关于如何使用匿名结构或联合编译 C 代码的问题,如C 具有使用联合的匿名字段。在 C 中,尝试使用包含匿名联合的命名结构创建类似的结构会导致编译错误。错误消息表明匿名联合和结构字段未在结构声明中声明。要在 C 中启用此功能,必须使用 -fms-extension...
    编程 发布于2024-11-08
  • 如何使用 OpenSSL 和 C++ 生成 SHA256 哈希值?
    如何使用 OpenSSL 和 C++ 生成 SHA256 哈希值?
    使用 OpenSSL 和 C 生成 SHA256 哈希 哈希是一种加密技术,用于生成数据的唯一指纹或摘要。对于 SHA256(安全哈希算法 2,256 位),此摘要是 256 位十六进制字符串。 SHA256 通常用于检查数据完整性、验证数字签名和安全存储密码。在本文中,我们将介绍如何使用 Open...
    编程 发布于2024-11-08
  • 探索软件工程师的就业市场
    探索软件工程师的就业市场
    Introduction In this article, we dive into the process of extracting and analyzing job data from LinkedIn, leveraging a combination of Python...
    编程 发布于2024-11-08
  • 在 React 中的选项卡之间发送数据。
    在 React 中的选项卡之间发送数据。
    本文将介绍如何在 React 全局组件之间发送数据,甚至在不同的浏览器选项卡中也是如此。 故事 想象您有一个项目列表,例如用户。 每个用户都可以在模态窗口中打开进行修改。 您没有任何后端订阅,这意味着如果用户发生变化,用户列表不会自动与后端同步。 因此,一旦用户的个人资料更新,您希望...
    编程 发布于2024-11-08
  • 如何从 WPF 中的非调度程序线程修改 ObservableCollection?
    如何从 WPF 中的非调度程序线程修改 ObservableCollection?
    “这种类型的 CollectionView 不支持从与调度程序线程不同的线程更改其 SourceCollection”问题描述A DataGrid 绑定异步填充的 ObservableCollection 会抛出错误,指出不允许从非 Dispatcher 线程对 SourceCollection 进...
    编程 发布于2024-11-08
  • 如何使用现代前端开发技术构建令人惊叹的网站
    如何使用现代前端开发技术构建令人惊叹的网站
    在当今的数字时代,网页设计在给访问者留下持久印象方面发挥着至关重要的作用。随着数以百万计的网站争夺注意力,创建一个令人惊叹的、脱颖而出的网站比以往任何时候都更加重要。现代前端开发技术彻底改变了网站的构建方式,使设计美观、实用且响应灵敏的网站变得更加容易,从而提供最佳的用户体验。本文将深入探讨可帮助您...
    编程 发布于2024-11-08
  • 让我们创建一个简单的 React hook 来检测浏览器及其功能
    让我们创建一个简单的 React hook 来检测浏览器及其功能
    用户代理嗅探是最流行的浏览器检测方法。不幸的是,由于多种原因,前端开发不太容易使用它。浏览器供应商不断尝试让嗅探变得不可能。因此,每个浏览器都有自己的用户代理字符串格式,解析起来非常复杂。 有一种更简单的方法可以使用浏览器 CSS API 实现相同的目的,我将向您展示。那么让我们创建浏览器功能检测 ...
    编程 发布于2024-11-08
  • 使用 Golang 的电子商务平台:了解干净的架构
    使用 Golang 的电子商务平台:了解干净的架构
    了解干净的架构 清洁架构(Clean Architecture)由 Robert C. Martin 推广,是一种软件设计理念,它将设计元素划分为环级别。干净架构的主要规则是代码依赖关系只能从外层向内移动。这意味着: 业务规则不依赖于 UI 或数据库。 业务规则对外界一无所知。 U...
    编程 发布于2024-11-08
  • TypeScript 与 JavaScript:开发人员的主要区别
    TypeScript 与 JavaScript:开发人员的主要区别
    JavaScript 是网络的核心语言,而 TypeScript 是基于它的现代增强语言。两者都很强大,但它们的用途略有不同。这是一个快速细分: 1. 类型安全 JavaScript:松散类型。变量可以动态更改类型,从而导致潜在的运行时错误。 TypeScript:静态类型。您定义类...
    编程 发布于2024-11-08
  • JavaScript 能否为不可预测的属性实现动态 Getter 和 Setter?
    JavaScript 能否为不可预测的属性实现动态 Getter 和 Setter?
    JavaScript 可以实现动态 Getters/Setters 吗?动态 getters 和 setters 允许 JavaScript 对象处理超出预定义属性的属性访问和修改。虽然早期的 JavaScript 技术对已知属性使用特定的 getter 和 setter,但本文探讨了为任何未定义的...
    编程 发布于2024-11-08
  • 我的第一个使用 Python 构建的开源项目,通过 CLI 快速检查数据库
    我的第一个使用 Python 构建的开源项目,通过 CLI 快速检查数据库
    我的问题是: 在处理其他项目时,我发现自己总是必须连接并使用 SELECT * 来查看虚拟条目或新用户。我更喜欢使用 CLI 来监视我的数据库条目,特别是因为我正在测试并只是将虚拟用户添加为项目中的第一个普通用户。因此,总是需要连接到 postgres、mysql 并从 CLI 进行 select ...
    编程 发布于2024-11-08
  • PHP,永不倒下的大象!
    PHP,永不倒下的大象!
    照片由 Ben Griffiths 在 Unsplash 上拍摄 PHP是一种广受好评的语言,同时也受到其他人的批评,有人说它正在消亡,但真的是这样吗,值得花时间学习PHP吗? PHP PHP 是 Rasmus Lerdorf 在 90 年代开发的一种编程语言,最初它被开发为一种服务...
    编程 发布于2024-11-08
  • 如何从 Android 应用程序安全访问远程 MySQL 数据库?
    如何从 Android 应用程序安全访问远程 MySQL 数据库?
    使用 JDBC 在 Android 中访问远程 MySQL 数据库:综合分析使用 JDBC API 从 Android 应用程序远程连接到 MySQL 数据库是一种常见的操作移动开发者之间的问题。虽然建立直接连接在技术上是可行的,但它带来了重大的安全和性能问题。安全影响允许 Android 应用程序...
    编程 发布于2024-11-08
  • 使用 CSS 创建自定义鼠标光标
    使用 CSS 创建自定义鼠标光标
    Written by Samson Omojola✏️ Editor’s note: This article was last updated by Njong Emy on 5 August 2024 to update content and code blocks, as well as t...
    编程 发布于2024-11-08

免责声明: 提供的所有资源部分来自互联网,如果有侵犯您的版权或其他权益,请说明详细缘由并提供版权或权益证明然后发到邮箱:[email protected] 我们会第一时间内为您处理。

Copyright© 2022 湘ICP备2022001581号-3