当前位置: 代码网 > it编程>编程语言>C/C++ > 实现 malloc() 和 free() — 分割大块

实现 malloc() 和 free() — 分割大块

2025年03月29日 C/C++ 我要评论
上一篇文章探讨了内存块重用顺序对内存消耗的影响,并优化了函数以减少浪费。然而,另一个更严重的问题依然存在:一个巨大的内存块可能会占据多个小块本可利用的空间。例如,分配一大块内存,释放后,再分配两个更小

实现 malloc() 和 free() — 分割大块

上一篇文章探讨了内存块重用顺序对内存消耗的影响,并优化了函数以减少浪费。然而,另一个更严重的问题依然存在:一个巨大的内存块可能会占据多个小块本可利用的空间。例如,分配一大块内存,释放后,再分配两个更小的块:

void *ptr1 = abmalloc(128);
void *ptr2 = abmalloc(8);
abfree(ptr1);
void *ptr3 = abmalloc(8);
void *ptr4 = abmalloc(8);
登录后复制

这时,128字节的空闲块无法被8字节的请求利用,导致后续8字节块分配需要再次扩展堆,造成内存利用率低下。

解决这个问题,一种高效但复杂的方法是使用“bins”:按大小分组的块列表。另一种更简单的方案是将大块分割成更小的块。本文采用后者。

代码重构

首先,对代码进行轻微重构。header_new() 函数同时负责分配内存和初始化块头,这不利于代码的可读性和维护性。我们将它拆分成两个函数:

  • header_plug():将已初始化的块插入到前一个和下一个块之间。
  • header_init():初始化块的元数据(大小和可用性)。

它们分别如下:

void header_init(header *header, size_t size, bool available) {
    header->size = size;
    header->available = available;
}

void header_plug(header *header, header *previous, header *next) {
    header->previous = previous;
    if (previous != null) {
        previous->next = header;
    }
    header->next = next;
    if (next != null) {
        next->previous = header;
    }
}
登录后复制

header_new() 函数修改如下:

header *header_new(header *previous, size_t size, bool available) {
    header *header = sbrk(sizeof(header) + size);
    header_init(header, size, available);
    header_plug(header, previous, null);
    return header;
}
登录后复制

(abmalloc() 函数中 last->previous->next = last; 这行可以删除,因为 header_plug() 现在负责处理此逻辑。)

分割内存块

接下来,实现 header_split() 函数。给定一个块头和所需最小大小,如果原始块足够大,则将其分割成两部分:

  • 所需大小的块;
  • 剩余部分及其新的块头;

首先,检查块是否足够大:

header *header_split(header *header, size_t size) {
    size_t original_size = header->size;
    if (original_size >= size + sizeof(header)) {
登录后复制

如果足够大,则分割块。首先,减小当前块的大小:

        header->size = original_size - size - sizeof(header);
登录后复制

计算新块的指针:

        header *new_header = (header + 1) + header->size; // corrected pointer calculation
登录后复制

初始化新块的头:

        header_init(new_header, size, true);
登录后复制

将新块连接到链表:

        header_plug(new_header, header, header->next);
登录后复制

如果原始块是最后一个块,更新 last 指针:

        if (header == last) {
            last = new_header;
        }
登录后复制

返回新块:

        return new_header;
    } else {
        return header;
    }
}
登录后复制

更新 abmalloc()

最后,修改 abmalloc() 函数,在找到可用块后,调用 header_split() 尝试分割它:

if (header->available && (header->size >= size)) {
    header = header_split(header, size);
    header->available = false;
    return (void*)(header + 1); // cast to void* for correct return type
}
登录后复制

如果块可以分割,则返回新块;否则,返回原始块。

关于块分割的说明

需要注意的是,新块是在原始块的末尾创建的。虽然也可以在开头创建,但在末尾创建新块可以使新的空闲块更靠近旧块,提高下次 abmalloc() 调用的效率。

分割大块内存是改进内存管理的一步,但它也可能导致小块内存碎片化,从而导致更大的请求需要扩展堆。下一篇文章将探讨如何解决这个问题。

以上就是实现 malloc() 和 free() — 分割大块的详细内容,更多请关注代码网其它相关文章!

(0)

相关文章:

版权声明:本文内容由互联网用户贡献,该文观点仅代表作者本人。本站仅提供信息存储服务,不拥有所有权,不承担相关法律责任。 如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 2386932994@qq.com 举报,一经查实将立刻删除。

发表评论

验证码:
Copyright © 2017-2025  代码网 保留所有权利. 粤ICP备2024248653号
站长QQ:2386932994 | 联系邮箱:2386932994@qq.com