【C 数据结构-动态内存管理】3. 伙伴系统管理动态内存

文章目录

  • 【 1. 伙伴系统的结构设计 】
  • 【 2. 分配算法 】
  • 【 3. 回收算法 】

  • 伙伴系统 本身是一种动态管理内存的方法,和边界标识法的区别是:使用伙伴系统管理的存储空间,无论是空闲块还是占用块,大小都是 2 的 n 次幂(n 为正整数)。
    例如,系统中整个存储空间为 2 m 2^m 2m 个字。那么在进行若干次分配与回收后,可利用空间表中只可能包含空间大小为: 2 0 2^0 20 2 1 2^1 21 2 2 2^2 22、…、 2 m 2^m 2m 的空闲块。

【 1. 伙伴系统的结构设计 】

  • 伙伴系统中可利用空间表中的结点构成如下图所示:
    header 域表示为头部结点,由 4 部分构成:
    • llinkrlink 为结点类型的指针域,分别用于指向直接前驱和直接后继结点。
    • tag :用于标记内存块的状态,是占用块(用 1 表示)还是空闲块(用 0 表示)
    • kval :记录该存储块的容量。由于系统中各存储块都是 2 的 m 幂次方,所以 kval 记录 m 的值。
      在这里插入图片描述
  • 伙伴系统节点的 C 实现:
typedef struct WORD_b{struct WORD_b *llink;//指向直接前驱int tag;//记录该块是占用块还是空闲块int kval;//记录该存储块容量大小为2的多少次幂struct WORD_b *rlink;//指向直接后继OtherType other;//记录结点的其它信息
}WORD_b,head;
  • 在伙伴系统中,由于系统会不断地接受用户的内存申请的请求,所以会产生很多大小不同但是容量都是为 2 m 2^m 2m 的内存块,所以为了在分配的时候查找方便,系统将大小相同的内存块建立一个链表。对于初始容量为 2 m 2^m 2m 的一整块存储空间来说,形成的链表就有可能有 m+1 个,为了更好的对这些链表进行管理,系统将这 m+1 个链表的表头存储在数组中,就类似于邻接表的结构,如下图所示:
    在这里插入图片描述
  • 可利用空间表的 C 实现:
#define m 16//设定m的初始值
typedef struct HeadNode {int nodesize;//记录该链表中存储的空闲块的大小WORD_b * first;//相当于链表中的next指针的作用
}FreeList[m+1];//一维数组

【 2. 分配算法 】

  • 伙伴系统的分配算法很简单。假设用户向系统申请大小为 n 的存储空间,若 2 k − 1 < n ≤ 2 k 2^{k-1} < n \leq 2^k 2k1<n2k,此时就需要查看可利用空间表中大小为 2 k 2^k 2k 的链表中有没有可利用的空间结点:
    • 如果该链表不为 NULL,可以直接采用头插法从头部取出一个结点,提供给用户使用;
    • 如果大小为 2k 的链表为 NULL,就需要依次查看比 2 k 2^k 2k 大的链表,找到后从链表中删除,截取相应大小的空间给用户使用,剩余的空间,根据大小插入到相应的链表中。
  • 例如,用户向系统申请一块大小为 7 个字的空间,而系统总的内存为 24 个字,则此时按照伙伴系统的分配算法得出: 2 2 2^2 22 < 7 < 2 3 2^3 23,所以此时应查看可利用空间表中大小为 2 3 2^3 23 的链表中是否有空闲结点:
    • 如果有,则从该链表中摘除一个结点,直接分配给用户使用;
    • 如果没有,则需依次查看比 2 3 2^3 23 大的各个链表中是否有空闲结点。假设,在大小 2 4 2^4 24 的链表中有空闲块,则摘除该空闲块,分配给用户 2 3 2^3 23 个字的空间,剩余 2 3 2^3 23 个字,该剩余的空闲块添加到大小为 2 3 2^3 23 的链表中,如下图所示:
      在这里插入图片描述
  • 使用伙伴系统进行存储空间的管理过程中,在用户申请空间时,由于大小不同的空闲块处于不同的链表中,所以 分配完成的速度会更快,算法相对简单

【 3. 回收算法 】

  • 无论使用什么内存管理机制,在内存回收的问题上都会面临一个共同的问题:如何把回收的内存进行有效地整合,伙伴系统也不例外。
  • 当用户申请的内存块不再使用时,系统需要将这部分存储块回收,回收时需要判断是否可以和其它的空闲块进行合并。在寻找合并对象时,伙伴系统和边界标识法不同,在伙伴系统中每一个存储块都有各自的 伙伴当用户释放存储块时只需要判断该内存块的伙伴是否为空闲块,如果是则将其合并,然后合并的新的空闲块还需要同其伙伴进行判断整合,反之直接将存储块根据大小插入到可利用空间表中即可
  • 判断一个存储块的伙伴的位置时,采用的方法为:如果该存储块的起始地址为 p,大小为 2 k 2^k 2k,则其伙伴所在的起始地址为:
    { p + 2 k ,若 p 取余 2 k + 1 = 0 p − 2 k ,若 p 取余 2 k + 1 = 2 k \begin{cases} p+2^k,若p取余2^{k+1}=0\\\\ p-2^k,若p取余2^{k+1}=2^k\end{cases} p+2k,若p取余2k+1=0p2k,若p取余2k+1=2k
  • 例如,当大小为 2 8 2^8 28 ,起始地址为 512 的伙伴块的起始地址的计算方式为:
    由于 512 取余 2 9 2^9 29=0,所以,512+ 2 8 2^8 28=768,即如果该存储块回收时,只需要查看起始地址为 768 768 768 的存储块的状态,如果是空闲块则两者合并,反之直接将回收的释放块链接到大小为 2 8 2^8 28 的链表中。
  • 回收存储空间时,对于 空闲块的合并,不是取决于该空闲块的相邻位置的块的状态;而是 完全取决于其伙伴块。所以即使其相邻位置的存储块时空闲块,但是由于两者不是伙伴的关系,所以也不会合并。这也就是该系统的缺点之一:由于 在合并时只考虑伙伴,所以容易产生存储的碎片

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

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【承装修、试电力施工许可证申报指南】广西承装(修、试)电力设施许可证资质代办流程

【承装修、试电力施工许可证申报指南】广西承装(修、试)电力设施许可证资质代办流程 广西承装(修、试)电力设施许可证资质代办流程 在广西地区&#xff0c;承装、修、试电力设施许可证的资质代办流程相对复杂&#xff0c;但遵循一定的步骤和规定&#xff0c;可以确保流程的顺利…

MobileNet网络详解

一、了解 网络亮点&#xff1a; 1、DW网络&#xff0c;大大减少运算量核参数数量 2、增加超参数&#xff1a;控制卷积层卷积核个数的超参数 &#xff0c;控制图像输入大小的超参数 &#xff0c;这两个超参数是人为设定的&#xff0c;不是机器学习到的。 二、DW卷积&#xff…

Windows环境编译 VVenC 源码生成 Visual Studio 工程

VVenC介绍 Fraunhofer通用视频编码器(VVenC)的开发是为了提供一种公开可用的、快速和有效的VVC编码器实现。VVenC软件基于VTM&#xff0c;其优化包括软件重新设计以减轻性能瓶颈、广泛的SIMD优化、改进的编码器搜索算法和基本的多线程支持以利用并行。此外&#xff0c;VVenC支…

端口被其他进程占用:OSError: [Errno 98] Address already in use

一、问题描述 错误提示端口号正在被使用 二、解决办法 1.使用 lsof 命令&#xff0c;列出所有正在监听&#xff08;即被绑定&#xff09;的网络连接&#xff0c;包括它们所使用的端口号 sudo lsof -i -P -n | grep LISTEN 2.解绑被绑定的端口号 根据 netstat 或 lsof 命令…

【C++】对文章分词,并对词频用不同排序方法排序,比较各排序算法效率

文章分词 1&#xff0e;问题描述2&#xff0e;需求分析3&#xff0e;概要设计3.1 主程序流程3.2 函数调用关系 4&#xff0e;主函数实现4.1 main.h4.2 main.cpp 5. 函数实现5.1 processDic函数5.2 forwardMax函数5.3 countWordFreq函数5.4 quickResult函数5.5 bubbleResult函数…

大模型系列之解读MoE

Mixtral 8x7B 的推出&#xff0c; 使我们开始更多地关注 基于MoE 的大模型架构&#xff0c; 那么&#xff0c;什么是MoE呢&#xff1f; 1. MoE溯源 MoE的概念起源于 1991 年的论文 Adaptive Mixture of Local Experts&#xff08;https://www.cs.toronto.edu/~hinton/absps/jjn…

【GDPU】数据结构实验十 哈夫曼编码

【实验内容】 1、假设用于通信的电文仅由8个字母 {a, b, c, d, e, f, g, h} 构成&#xff0c;它们在电文中出现的概率分别为{ 0.07, 0.19, 0.02, 0.06, 0.32, 0.03, 0.21, 0.10 }&#xff0c;试为这8个字母设计哈夫曼编码。 提示:包含两个过程:&#xff08;1&#xff09;构建…

C语言实现面向对象—以LED驱动为例

点亮一个LED 常见的LED代码 分层分离思想 面向对象的LED驱动 LED左边高电平。 当LED右边为低电平时&#xff0c;LED有电流通过&#xff0c;LED亮。反之&#xff0c;LED灭 GPIO功能描述&#xff1a; 点亮LED的步骤及代码&#xff1a; 开启GPIO的时钟 配置GPIO为输出模式 …

04-22 周日 阿里云-瑶光上部署FastBuild过程(配置TLS、自定义辅助命令)

04-22 周日 阿里云-瑶光上部署FastBuild过程 时间版本修改人描述2024年4月22日14:18:59V0.1宋全恒新建文档2024年4月23日20:41:26V1.0宋全恒完成了基本流程的添加 简介 前提 准备两台服务&#xff0c;一台部署Docker&#xff0c;一台部署FastBuild的镜像容器服务所述的Docke…

stm32f103zet6_DAC_2_输出电压

实现效果 DAC输出的电压 同过电压表测量电压 1.DAC配置的步骤 初始化DAC时钟。配置DAC的GPIO端口。设置DAC的工作模式&#xff08;例如&#xff0c;是否使用触发功能&#xff0c;是否启用DAC中断等&#xff09;。启动DAC。 2常用的函数 函数 HAL_DAC_Start() - 开启指定…

软件系统概要设计说明书(实际项目案例整理模板套用)

系统概要设计说明书 1.整体架构 2.功能架构 3.技术架构 4.运行环境设计 5.设计目标 6.接口设计 7.性能设计 8.运行设计 9.出错设计 全文档获取进主页 软件资料清单列表部分文档&#xff08;全套可获取&#xff09;&#xff1a; 工作安排任务书&#xff0c;可行性分析报告&…

阿里巴巴alibaba国际站API接口:商品详情和关键词搜索商品列表

阿里巴巴国际站&#xff08;Alibaba.com&#xff09;提供了API接口供开发者使用&#xff0c;以实现与平台的数据交互。然而&#xff0c;由于API的详细内容和调用方式可能会随着时间和平台更新而发生变化&#xff0c;以下是一个概述和一般性的指导&#xff0c;关于如何使用阿里巴…