当前位置: 首页 > news >正文

33. 简易内存池

1、题目描述
● 请实现一个简易内存池,根据请求命令完成内存分配和释放。
● 内存池支持两种操作命令,REQUEST和RELEASE,其格式为:
● REQUEST=请求的内存大小 表示请求分配指定大小内存,如果分配成功,返回分配到的内存首地址;如果内存不足,或指定的大小为0,则输出error。
● RELEASE=释放的内存首地址 表示释放掉之前分配的内存,释放成功无需输出,如果释放不存在的首地址则输出error。
注意:

内存池总大小为100字节。
内存池地址分配必须是连续内存,并优先从低地址分配。
内存释放后可被再次分配,已释放的内存在空闲时不能被二次释放。
不会释放已申请的内存块的中间地址。
释放操作只是针对首地址所对应的单个内存块进行操作,不会影响其它内存块。
2、输入描述
首行为整数 N , 表示操作命令的个数,取值范围:0 < N <= 100。
接下来的N行, 每行将给出一个操作命令,操作命令和参数之间用 “=”分割。

3、输出描述
请求分配指定大小内存时,如果分配成功,返回分配到的内存首地址;如果内存不足,或指定的大小为0,则输出error
释放掉之前分配的内存时,释放成功无需输出,如果释放不存在的首地址则输出error。
用例:

输入
5
REQUEST=10
REQUEST=20
RELEASE=0
REQUEST=20
REQUEST=10

输出
0
10
30
0

ps:
第一条,申请地址0~9的10个字节内存,返回首地址0
第二条,申请地址10~29的20字节内存,返回首地址10
第三条,释放首地址为0的内存申请,0~9地址内存被释放,变为空闲,释放成功,无需输出
第四条,申请20字节内存,09地址内存连续空间不足20字节,往后查找到3049地址,返回首地址30
第五条,申请10字节,0~9地址内存空间足够,返回首地址0
————————————————

                            版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
                        
原文链接:https://blog.csdn.net/weixin_52914380/article/details/138459806

一、问题分析

首先读题,仔细看描述中的内容,发现需求是

1.请实现一个简易内存池,根据请求命令完成内存分配和释放

2.内存池支持两种操作命令,REQUEST和RELEASE,其格式为:

REQUEST=请求的内存大小,表示请求分配指定大小内存,如果分配成功,返回分配到的内存首地址;如果内存不足,或指定的大小为0,则输出error。

RELEASE=释放的内存首地址,表示释放掉之前分配的内存,释放成功无需输出,如果释放不存在的首地址则输出error。

3.注意:(1)内存池总大小为100字节。

(2)内存池地址分配必须是连续内存,并优先从低地址分配

(3)内存释放后可被再次分配,已释放的内存在空闲时不能被二次释放

(4)不会释放已申请的内存块的中间地址

(5)释放操作只是针对首地址所对应的单个内存块进行操作,不会影响其他内存块。

4.输入描述:首行为整数N,表示操作命令的个数,取值范围:N大于0小于等于100.

接下来的N行,每行将给出一个操作命令,操作命令和参数之间用“=”分割。

5.输出描述:请求分配指定大小内存时,如果分配成功,返回分配到的内存首地址;如果内存不足,或指定的大小为0,则输出error

释放掉之前分配的内存时,释放成功无需输出,如果释放不存在的首地址则输出error。

二、解题思路

1.内存池一共有两种命令,一种是REQUEST,后面跟请求分配的内存大小

一种是RELEASE后面跟要释放内存的首地址

2.首先我们引入标准输入输出库、标准库、字符串处理库定义内存池总大小为100字节

#include <stdio.h>

#include <stdlib.h>

#include <string.h>

#define MEMORY_POOL_SIZE 100

3.然后定义MemoryBlock结构体用来表示内存池中的内存块信息。其中startAddress记录内存块的起始地址,size表示内存块的大小(字节数),isAllocated用于标记该内存块是否已经被分配出去(1表示已分配,0表示空闲),next指针用于链接下一个内存块,形成一个链表结构,方便管理内存池中多个内存块的情况。

typede struct MemoryBlock {

int startAddress; // 内存块起始地址

int size; // 内存块大小

int isAllocated; // 内存块是否已分配

struct MemoryBlock *next; // 指向下一个内存块的指针

} MemoryBlock;

4.然后声明一个函数用于创建并初始化内存池,通过动态分配内存创建一个MemoryBlock结构体来表示整个内存池,将其起始地址设为0,大小设为定义好的MEMORY_POOL_SIZE(也就是100字节),标记为未分配状态(isAllocated设为0),并将next指针设为NULL,表示初始时只有这一个代表整个内存池的空闲内存块,最后返回这个表示内存池的结构体指针。

MemoryBlock* initMemoryPool() {

MemoryBlock *pool = (MemoryBlock *)malloc(sizeof(MemoryBlock));

if(pool == NULL) {

perror("内存分配失败");

return NULL;

}

pool->startAddress = 0;

pool->size = MEMORY_POOL_SIZE;

pool->isAllocated = 0;

pool->next = NULL;

return pool;

}

5.声明一个函数用来查找空闲内存块,该函数接收内存池的头指针pool和请求分配的内存大小requestSize作为参数,通过遍历内存池链表(从内存池的头指针开始,沿着next指针逐个访问内存块),查找是否存在未分配(isAllocated为0)且大小足够(size大于等于requestSize)的空闲内存块,找到则返回该内存块的指针,若遍历完整个链表都没有找到符合条件的空闲内存块,就返回NULL。

MemoryBlock* findFreeBlock(MemoryBlock *pool, int requestSize) {

MemoryBlock *current = pool;

while(current != NULL) {

if(current->isAllocated == 0 && current->size >= requestSize) {

return current;

}

current = current->next;

}

return NULL;

}

6.声明一个内存分配函数用于实际分配内存,首先判断请求分配的内存大小是否为0,若是则输出error并返回-1表示分配失败。然后调用findFreeBlock函数在内存池中查找合适的空闲内存块,如果没找到(返回NULL),同样输出error并返回-1.若找到了空闲内存块,分两种情况处理:

如果找到的空闲内存块大小正好等于请求的大小,直接将该内存块的isAllocated标记为1,表示已分配,然后返回该内存块的起始地址。

如果空闲内存块大小大于请求的大小,需要将其拆分成两个内存块,一个用于分配(大小设为requestSize,标记为已分配),另一个保持空闲(大小为原空闲内存块大小减去请求的大小,标记为未分配),并通过调整链表指针将新的空闲内存块插入到链表中合适的位置,最后返回分配的内存块的起始地址。

int allocateMemory(MemoryBlock **pool, int requestSize) {

if(requestSize == 0) {

printf("error\n");

return -1;

}

MemoryBlock *freeBlock = findFreeBlock(*pool, requestSize);

if(freeBlock == NULL) {

printf("error\n");

return -1;

}

if(freeBlock->size == requestSize) {

freeBlock->isAllocated = 1;

} else {

MemoryBlock *newBlock = (MemoryBlock *)malloc(sizeof(MemoryBlock));

if(newBlock == NULL) {

perror("内存分配失败");

return -1;

}

newBlock->startAddress = freeBlock->startAddress + requestSize;

newBlock->size = freeBlock->size - requestSize;

newBlock->isAllocated = 0;

newBlock->next = freeBlock->next;

freeBlock->size = requestSize;

freeBlock->isAllocated = 1;

freeBlock->next = newBlock;

}

return freeBlock->startAddress;

}

7.然后是释放内存函数,用于释放指定首地址对应的内存块,通过遍历内存池链表来查找要释放的内存块(根据起始地址匹配),如果找到了对应的内存块且该内存块是已分配状态,则将其标记为未分配,并且和前后有可能存在的空闲内存块合并,

void releaseMemory(MemoryBlock **pool, int relaseAddress) {

MemoryBlock *prev = NULL;

MemoryBlock *current = *pool;

while(current != NULL) {

if(current->startAddress == releaseAddress) {

if(current->isAllocated == 0) {

printf("error\n");

return;

}

current->isAllocated = 0;

// 如果前面有空闲内存块我们尝试合并

if(prev != NULL && prev->isAllocated == 0) {

prev->size += current->size;

prev->next = current->next;

free(current);

current = prev;

}

// 如果后面有空闲内存块,我们尝试合并

if(current->next != NULL && current->next->isAllocated == 0) {

current->size += current->next->size;

current->next = current->next->next;

}

return;

}

prev = current;

current = current->next;

}

printf("error\n");

}

8.主函数首先定义一个整数int N;用于读取命令的个数

int main() {

int N;

scanf("%d", &N);

MemoryBlock *memoryPool = initMemoryPool();

for(int i = 0;i < N; i++) {

char command[20];

scanf("%s",command);

if(strncmp(command,"REQUEST", 7) == 0) {

int requestSize;

sscanf(command + 7, "=%d", &requestSize);

int address = allocateMemory(&memoryPool, requestSize);

if(address != -1) {

printf("%d\n", address);

}

}

else if(strncmp(command,"RELEASE", 7) == 0) {

int relaseAddress;

sscanf(command + 7, "=%d", &releaseAddress);

releaseMemory(&memoryPool, releaseAddress);

}

}

return 0;

}

三、具体步骤

使用的语言是C

#include <stdio.h>
#include <stdlib.h>
#include <stdbool.h>
#include <string.h>
#define MEMORY_POOL_SIZE 100// 内存块结构体
typedef struct MemoryBlock {int startAddress;int size;bool isAllocated;struct MemoryBlock *next;
} MemoryBlock;// 初始化内存池
MemoryBlock* initMemoryBlock() {MemoryBlock *pool = (MemoryBlock *)malloc(sizeof(MemoryBlock));if(pool == NULL) {perror("内存分配失败");return NULL;}pool->startAddress = 0;pool->size = MEMORY_POOL_SIZE;pool->isAllocated = false;pool->next = NULL;return pool;
}// 寻找空闲内存池
MemoryBlock* findFreeBlock(MemoryBlock* pool, int requestSize) {MemoryBlock *current = pool;while(current != NULL) {if(current->isAllocated == false && current->size >= requestSize) {return current;}current = current->next;}return NULL;
}// 为空闲地址池分配内存
int allocateMemory(MemoryBlock** pool, int requestSize) {if(requestSize == 0) {printf("error\n");return -1;}MemoryBlock *freeBlock = findFreeBlock(*pool, requestSize);if(freeBlock == NULL) {printf("error\n");return -1;}if(freeBlock->size == requestSize) {freeBlock->isAllocated = true;} else {MemoryBlock *newBlock = (MemoryBlock*)malloc(sizeof(MemoryBlock));if(newBlock == NULL) {perror("内存分配失败");return -1;}newBlock->size = freeBlock->size - requestSize;newBlock->isAllocated = false;newBlock->startAddress = freeBlock->startAddress + requestSize;newBlock->next = freeBlock->next;freeBlock->isAllocated = true;freeBlock->size = requestSize;freeBlock->next = newBlock;}return freeBlock->startAddress;
}void releaseMemory(MemoryBlock** pool, int relaseAddress) {MemoryBlock* prev = NULL, *current = *pool;while(current != NULL) {if(current->startAddress == relaseAddress) {if(current->isAllocated == false) {printf("error\n");return;}current->isAllocated = false;if(prev!= NULL && prev->isAllocated == false) {prev->size += current->size;prev->next = current->next;free(current);current = prev;}if(current->next != NULL && current->next->isAllocated == 0) {current->size += current->next->size;current->next = current->next->next;}return;}prev = current;current = current->next;}printf("error\n");
}int main() {int N;scanf("%d", &N);MemoryBlock *memoryPool = initMemoryBlock();for(int i = 0; i < N; i++) {char command[20];int num;scanf("%s", command);char *temp;temp = strtok(command, "=");num = atoi(strtok(NULL, "="));//printf("%s=%d\n", temp, num);if(strcmp(temp,"REQUEST") == 0) {int address = allocateMemory(&memoryPool, num);if(address != -1) {printf("%d\n", address);}} else if(strcmp(temp, "RELEASE") == 0) {releaseMemory(&memoryPool, num);}}return 0;
}

相关文章:

33. 简易内存池

1、题目描述 ● 请实现一个简易内存池,根据请求命令完成内存分配和释放。 ● 内存池支持两种操作命令&#xff0c;REQUEST和RELEASE&#xff0c;其格式为&#xff1a; ● REQUEST请求的内存大小 表示请求分配指定大小内存&#xff0c;如果分配成功&#xff0c;返回分配到的内存…...

win32汇编环境,对话框程序模版,含文本框与菜单简单功能

;运行效果 ;win32汇编环境,对话框程序模版&#xff0c;含文本框与菜单简单功能 ;直接抄进RadAsm可编译运行。 ;下面为asm文件 ;>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>&g…...

人工智能与传统编程的主要区别是什么?

传统编程&#xff1a;开发者预先编写软件行为规则&#xff0c;代码基于程序员定义逻辑处理输入并产生确定输出&#xff0c;具有确定性、手动编写规则和结构化逻辑特点&#xff0c;如垃圾邮件分类程序基于预设关键词等规则。AI 编程&#xff1a;从数据中学习而非手动编写规则&am…...

实战交易策略 篇十一:一揽子交易策略

文章目录 系列文章适用条件核心策略小额大量投资行业或主题聚焦同步操作优势系列文章 实战交易策略 篇一:奥利弗瓦莱士短线交易策略 实战交易策略 篇二:杰西利弗莫尔股票大作手操盘术策略 实战交易策略 篇三:333交易策略 实战交易策略 篇四:价值投资交易策略 实战交易策略…...

doris 2.1 -Data Manipulation-Transaction

注意:doris 只能控制读一致性,并不能rollback 1 Explicit and Implicit Transactions 1.1 Explicit Transactions 1.1.1 Explicit transactions require users to actively start, commit transactions. Only insert into values statement is supported in 2.1. BEGIN; …...

多模态融合:阿尔茨海默病检测

&#x1f368; 本文为&#x1f517;365天深度学习训练营 中的学习记录博客&#x1f356; 原作者&#xff1a;K同学啊 一、实验介绍 本实验包含 645 名阿尔茨海默病受试者&#xff0c;分为 AD、CN 和 MCI 组&#xff0c;数据集包含 3D MRI 图像与一份CSV数据&#xff0c;MRI数据…...

Ceph 手动部署(CentOS9)

#Ceph手动部署、CentOS9、squid版本、数字版本19.2.0 #部署服务:块、对象、文件 一、部署前规划 1、兼容性确认 2、资源规划 节点类型节点名称操作系统CPU/内存硬盘网络组件安装集群节点CephAdm01CentOS94U/8GOS:40G,OSD:2*100GIP1:192.169.0.9(管理&集群),IP2:…...

家政预约小程序05活动管理

目录 1 搭建活动管理页面2 搭建活动规则页面3 搭建规则新增页面3 配置规则跳转4 搭建活动参与记录总结 上一篇我们介绍了活动管理的表结构设计&#xff0c;本篇我们介绍一下后台功能。 1 搭建活动管理页面 我们一共搭建了三个表&#xff0c;先搭建主表的后台功能。打开我们的后…...

解决安装pynini和WeTextProcessing报错问题

点击这里&#xff0c;访问博客 0. 背景 最近在给别人有偿部署ASR-LLM-TTS项目时遇到安装pynini和WeTextProcessing依赖报错的问题&#xff0c;报错信息如下&#xff1a; IC:\Program Files (x86)\Windows Kits\10\include\10.0.22621.0\ucrt" "-IC:\Program Files…...

【PCIe 总线及设备入门学习专栏 4.1 -- PCI 总线的地址空间分配】

文章目录 Overview 本文转自&#xff1a;https://blog.chinaaet.com/justlxy/p/5100053219 Overview PCI 总线具有32位数据/地址复用总线&#xff0c;所以其存储地址空间为 2324GB。也就是PCI上的所有设备共同映射到这4GB上&#xff0c;每个PCI设备占用唯一的一段PCI地址&…...

华为配置 之 RIP

简介&#xff1a; RIP&#xff08;路由信息协议&#xff09;是一种广泛使用的内部网关协议&#xff0c;基于距离向量算法来决定路径。它通过向全网广播路由控制信息来动态交换网络拓扑信息&#xff0c;从而计算出最佳路由路径。RIP易于配置和理解&#xff0c;非常适用于小型网络…...

探寻AI Agent:开启知识图谱自动生成新篇章(17/30)

一、AI Agent 与知识图谱&#xff1a;智能时代的双雄 在当今科技飞速发展的时代&#xff0c;人工智能如同一股汹涌澎湃的浪潮&#xff0c;正以前所未有的力量重塑着我们的世界。而在这股浪潮中&#xff0c;AI Agent 与知识图谱无疑是两颗最为璀璨的明珠&#xff0c;它们各自发挥…...

卸载wps后word图标没有变成白纸恢复

这几天下载了个wps教育版&#xff0c;后头用完了删了 用习惯的2019图标 给兄弟我干没了&#xff1f;&#xff1f;&#xff1f; 其他老哥说什么卸载关联重新下 &#xff0c;而且还要什么撤销保存原来的备份什么&#xff0c;兄弟也是不得不怂了 后头就发现了这个半宝藏博主&…...

LeetCode 热题 100_二叉树的直径(40_543_简单_C++)(二叉树;递归)

LeetCode 热题 100_二叉树的直径&#xff08;40_543&#xff09; 题目描述&#xff1a;输入输出样例&#xff1a;题解&#xff1a;解题思路&#xff1a;思路一&#xff08;递归&#xff09;&#xff1a; 代码实现代码实现&#xff08;思路一&#xff08;递归&#xff09;&#…...

【数据结构】线性数据结构——链表

1. 定义 链表是一种线性数据结构&#xff0c;由多个节点&#xff08;Node&#xff09;组成。每个节点存储数据和指向下一个节点的指针。与数组不同&#xff0c;链表的节点不需要在内存中连续存储。 2. 特点 动态存储&#xff1a; 链表的大小不固定&#xff0c;可以动态增加或…...

开源存储详解-分布式存储与ceph

ceph体系结构 rados&#xff1a;reliable, autonomous, distributed object storage, rados rados采用c开发 对象存储 ceph严格意义讲只提供对象存储能力&#xff0c;ceph的块存储能力实际是基于对象存储库librados的rbd 对象存储特点 对象存储采用put/get/delete&#xf…...

[算法] [leetcode-509] 斐波那契数

509 斐波那契数 斐波那契数 &#xff08;通常用 F(n) 表示&#xff09;形成的序列称为 斐波那契数列 。该数列由 0 和 1 开始&#xff0c;后面的每一项数字都是前面两项数字的和。也就是&#xff1a; F(0) 0&#xff0c;F(1) 1 F(n) F(n - 1) F(n - 2)&#xff0c;其中 n…...

运维人员的Go语言学习路线

以下是一份更为详细的适合运维人员的Go语言学习路线图&#xff1a; 一、基础环境搭建与入门&#xff08;第 1 - 2 周&#xff09; 第 1 周 环境搭建 在本地开发机和常用的运维服务器环境&#xff08;如 Linux 系统&#xff09;中安装 Go 语言。从官方网站&#xff08;https://…...

[创业之路-222]:波士顿矩阵与GE矩阵在业务组合选中作用、优缺点比较

目录 一、波士顿矩阵 1、基本原理 2、各象限产品的定义及战略对策 3、应用 4、优点与局限性 二、技术成熟度模型与产品生命周期模型的配对 1、技术成熟度模型 2、产品生命周期模型 3、技术成熟度模型与产品生命周期模型的配对 三、产品生命周期与产品类型的对应关系 …...

安卓入门十一 常用网络协议四

MQTT&#xff08;Message Queuing Telemetry Transport&#xff09; MQTT是一种轻量级的、发布/订阅模式的消息传输协议。它被设计用于在低带宽或不稳定网络环境下&#xff0c;实现物联网设备之间的可靠通信。 4.1 MQTT详细介绍 发布/订阅模式&#xff1a;MQTT 使用发布/订…...

《机器学习》——利用OpenCV库中的KNN算法进行图像识别

文章目录 KNN算法介绍下载OpenCV库实验内容实验结果完整代码手写数字传入模型训练 KNN算法介绍 一、KNN算法的基本要素 K值的选择&#xff1a;K值代表选择与新测试样本距离最近的前K个训练样本数&#xff0c;通常K是不大于20的整数。K值的选择对算法结果有重要影响&#xff0c…...

StarRocks 存算分离在得物的降本增效实践

编者荐语&#xff1a; 得物优化数据引擎布局&#xff0c;近期将 4000 核 ClickHouse 迁移至自建 StarRocks&#xff0c;成本降低 40%&#xff0c;查询耗时减半&#xff0c;集群稳定性显著提升。本文详解迁移实践与成果&#xff0c;文末附丁凯剑老师 StarRocks Summit Asia 2024…...

Tube Qualify弯管测量系统在汽车管路三维检测中的应用

从使用量上来说&#xff0c;汽车行业是使用弯管零件数量最大的单一行业。在汽车的燃油&#xff0c;空调&#xff0c;排气&#xff0c;转向&#xff0c;制动等系统中都少不了管路。汽车管件形状复杂&#xff0c;且由于安装空间限制&#xff0c;汽车管件拥有不同弯曲半径&#xf…...

udp分片报文发送和接收

读文件通过udp分片发送的目的端&#xff1a;&#xff08;包含错误的分片包&#xff09; #!/usr/bin/python # -*- coding: utf-8 -*-#python send_100frag_file.py -p 55432 -f snatdownloadimport argparse import loggingfrom scapy.all import *# Define the maximum size …...

【从零开始入门unity游戏开发之——C#篇39】C#反射使用——Type 类、Assembly 类、Activator 类操作程序集

文章目录 前言一、前置知识1、编译器2、程序集&#xff08;Assembly&#xff09;3、元数据&#xff08;Metadata&#xff09; 二、反射1、反射的概念2、反射的作用3、反射的核心Type 类3.1 Type 类介绍3.2 不同方法获取 Type3.3 获取type类型所在的程序集的相关信息 4、反射的常…...

安卓触摸事件的传递

setOnTouchListener()返回值的副作用&#xff08;触摸事件是否继续往下或往后传递&#xff09;如下&#xff1a; 返回值效果是否往下层view传递是否往当前view的后续监听传递true该pointer离开屏幕前的后续所有触摸事件都会传递给该TouchListener否否false该pointer离开屏幕前…...

idea项目导入gitee 码云

1、安装gitee插件 IDEA 码云插件已由 gitosc 更名为 gitee。 1 在码云平台帮助文档http://git.mydoc.io/?t153739上介绍的很清楚&#xff0c;推荐前两种方法&#xff0c; 搜索码云插件的时候记得名字是gitee&#xff0c;gitosc已经搜不到了。 2、使用码云托管项目 如果之…...

典型常见的基于知识蒸馏的目标检测方法总结三

来源&#xff1a;Google学术2023-2024的顶会顶刊论文 NeurIPS 2022&#xff1a;Towards Efficient 3D Object Detection with Knowledge Distillation 为3D目标检测提出了一种知识蒸馏的Benchmark范式&#xff0c;包含feature的KD&#xff0c;Logit的cls和reg的KD&#xff0c…...

端口被占用

端口8080被占用 哈哈哈&#xff0c;我是因为后端项目跑错了&#xff0c;两个项目后端名称太像了&#xff1b; &#xff08;1&#xff09;netstat -aon | findstr 8080&#xff0c;找到占用8080端口的进程号&#xff0c;获取对应的进程号pid&#xff1b; &#xff08;2&#…...

Javascript知识框架图(待完善)

以下是一个清晰且详细的 JavaScript 知识框架&#xff0c;涵盖基础知识到高级概念&#xff0c;适合学习和参考&#xff1a; JavaScript 知识框架 1. 基础知识 数据类型 原始类型&#xff1a;Number&#xff0c;String&#xff0c;Boolean&#xff0c;Null&#xff0c;Undefin…...