」工欲善其事,必先利其器。「—孔子《論語.錄靈公》
首頁 > 程式設計 > 實作 malloc() 和 free() — 首先重複使用舊內存

實作 malloc() 和 free() — 首先重複使用舊內存

發佈於2024-11-08
瀏覽:123

在本系列关于实现 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]刪除
最新教學 更多>
  • 我可以將加密從McRypt遷移到OpenSSL,並使用OpenSSL遷移MCRYPT加密數據?
    我可以將加密從McRypt遷移到OpenSSL,並使用OpenSSL遷移MCRYPT加密數據?
    將我的加密庫從mcrypt升級到openssl 問題:是否可以將我的加密庫從McRypt升級到OpenSSL?如果是這樣,如何? 答案:是的,可以將您的Encryption庫從McRypt升級到OpenSSL。 可以使用openssl。 附加說明: [openssl_decrypt()函數要求...
    程式設計 發佈於2025-05-03
  • PHP與C++函數重載處理的區別
    PHP與C++函數重載處理的區別
    作為經驗豐富的C開發人員脫離謎題,您可能會遇到功能超載的概念。這個概念雖然在C中普遍,但在PHP中構成了獨特的挑戰。讓我們深入研究PHP功能過載的複雜性,並探索其提供的可能性。 在PHP中理解php的方法在PHP中,函數超載的概念(如C等語言)不存在。函數簽名僅由其名稱定義,而與他們的參數列表無關...
    程式設計 發佈於2025-05-03
  • 圖片在Chrome中為何仍有邊框? `border: none;`無效解決方案
    圖片在Chrome中為何仍有邊框? `border: none;`無效解決方案
    在chrome 在使用Chrome and IE9中的圖像時遇到的一個頻繁的問題是圍繞圖像的持續薄薄邊框,儘管指定了圖像,儘管指定了;和“邊境:無;”在CSS中。要解決此問題,請考慮以下方法: Chrome具有忽略“ border:none; none;”的已知錯誤,風格。要解決此問題,請使用以下...
    程式設計 發佈於2025-05-03
  • 在PHP中如何高效檢測空數組?
    在PHP中如何高效檢測空數組?
    在PHP 中檢查一個空數組可以通過各種方法在PHP中確定一個空數組。如果需要驗證任何數組元素的存在,則PHP的鬆散鍵入允許對數組本身進行直接評估:一種更嚴格的方法涉及使用count()函數: if(count(count($ playerList)=== 0){ //列表為空。 } 對...
    程式設計 發佈於2025-05-03
  • 在Python中如何創建動態變量?
    在Python中如何創建動態變量?
    在Python 中,動態創建變量的功能可以是一種強大的工具,尤其是在使用複雜的數據結構或算法時,Dynamic Variable Creation的動態變量創建。 Python提供了幾種創造性的方法來實現這一目標。 利用dictionaries 一種有效的方法是利用字典。字典允許您動態創建密鑰並...
    程式設計 發佈於2025-05-03
  • 表單刷新後如何防止重複提交?
    表單刷新後如何防止重複提交?
    在Web開發中預防重複提交 在表格提交後刷新頁面時,遇到重複提交的問題是常見的。要解決這個問題,請考慮以下方法: 想像一下具有這樣的代碼段,看起來像這樣的代碼段:)){ //數據庫操作... 迴聲“操作完成”; 死(); } ? > ...
    程式設計 發佈於2025-05-03
  • 如何使用FormData()處理多個文件上傳?
    如何使用FormData()處理多個文件上傳?
    )處理多個文件輸入時,通常需要處理多個文件上傳時,通常是必要的。 The fd.append("fileToUpload[]", files[x]); method can be used for this purpose, allowing you to send multi...
    程式設計 發佈於2025-05-03
  • 如何使用組在MySQL中旋轉數據?
    如何使用組在MySQL中旋轉數據?
    在關係數據庫中使用mySQL組使用mySQL組進行查詢結果,在關係數據庫中使用MySQL組,轉移數據的數據是指重新排列的行和列的重排以增強數據可視化。在這裡,我們面對一個共同的挑戰:使用組的組將數據從基於行的基於列的轉換為基於列。 Let's consider the following ...
    程式設計 發佈於2025-05-03
  • 如何使用Regex在PHP中有效地提取括號內的文本
    如何使用Regex在PHP中有效地提取括號內的文本
    php:在括號內提取文本在處理括號內的文本時,找到最有效的解決方案是必不可少的。一種方法是利用PHP的字符串操作函數,如下所示: 作為替代 $ text ='忽略除此之外的一切(text)'; preg_match('#((。 &&& [Regex使用模式來搜索特...
    程式設計 發佈於2025-05-03
  • Java數組中元素位置查找技巧
    Java數組中元素位置查找技巧
    在Java數組中檢索元素的位置 利用Java的反射API將數組轉換為列表中,允許您使用indexof方法。 (primitives)(鏈接到Mishax的解決方案) 用於排序陣列的數組此方法此方法返回元素的索引,如果發現了元素的索引,或一個負值,指示應放置元素的插入點。
    程式設計 發佈於2025-05-03
  • eval()vs. ast.literal_eval():對於用戶輸入,哪個Python函數更安全?
    eval()vs. ast.literal_eval():對於用戶輸入,哪個Python函數更安全?
    稱量()和ast.literal_eval()中的Python Security 在使用用戶輸入時,必須優先確保安全性。強大的Python功能Eval()通常是作為潛在解決方案而出現的,但擔心其潛在風險。 This article delves into the differences betwee...
    程式設計 發佈於2025-05-03
  • C++20 Consteval函數中模板參數能否依賴於函數參數?
    C++20 Consteval函數中模板參數能否依賴於函數參數?
    [ consteval函數和模板參數依賴於函數參數在C 17中,模板參數不能依賴一個函數參數,因為編譯器仍然需要對非contexexpr futcoriations contim at contexpr function進行評估。 compile time。 C 20引入恆定函數,必須在編譯時進...
    程式設計 發佈於2025-05-03
  • 找到最大計數時,如何解決mySQL中的“組函數\”錯誤的“無效使用”?
    找到最大計數時,如何解決mySQL中的“組函數\”錯誤的“無效使用”?
    如何在mySQL中使用mySql 檢索最大計數,您可能會遇到一個問題,您可能會在嘗試使用以下命令:理解錯誤正確找到由名稱列分組的值的最大計數,請使用以下修改後的查詢: 計數(*)為c 來自EMP1 按名稱組 c desc訂購 限制1 查詢說明 select語句提取名稱列和每個名稱...
    程式設計 發佈於2025-05-03
  • 在Go語言中如何簡潔定義10的冪常量
    在Go語言中如何簡潔定義10的冪常量
    在GO 利用浮點線文字一種簡潔的方式是使用浮點文字,該方法是使用floingpoint protals。寫作1E3比寫作1000更有效。這是一個示例(67個沒有空間的字符):的文字用於未構圖的整數常數,我們可以將1000用於KB,並用KB將隨後的常量乘以KB,如下所示(77個沒有空格的字符):,作...
    程式設計 發佈於2025-05-03

免責聲明: 提供的所有資源部分來自互聯網,如果有侵犯您的版權或其他權益,請說明詳細緣由並提供版權或權益證明然後發到郵箱:[email protected] 我們會在第一時間內為您處理。

Copyright© 2022 湘ICP备2022001581号-3