内存池学习(一)

article/2025/8/10 2:09:53

一、内存池

1、内存池所使用的内存是什么内存?

指的是虚拟内存(堆空间),而不是物理内存
在这里插入图片描述

2、为什么会有内存池?

一个系统或者程序长期运行,突然会coredump掉,并且程序又频繁地分配和释放内存,那么大概率就是内存碎片导致coredump。

3、什么是内存碎片?

内存动态分配和释放过程中产生的‌不连续空闲区域‌,其本质是未被高效利用的存储空间。

  • 外部碎片:空闲区域的总大小足够,但因为过小或者太过离散,导致没有一个单独的连续空闲区域能够满足当前分配请求。
  • 内部碎片:空闲区域的总大小大于当前分配请求的大小,但是分配的区域比实际需要的要大。比如:系统一页为4K,进程申请3K,剩余1K就未充分利用,称为内部碎片。

3、内存池的优点

  1. 减少内存碎片
  2. 提高内存分配效率,因为不需要每次都调用系统函数进行内存申请和释放。
  3. 减少内存泄漏的可能性

二、具体实现

1、定长

1、以一页为4KB为例。

每个内存块大小为x,那么一页就有y个内存块。
y = 4096 / x;

typedef struct mempool_s {int block_size; //每个内存块的大小int free_count; //空闲内存块的数量char* mem_ptr;  //指向内存池的起始地址char* free_list; //指向空闲内存块的链表头
}mempool;
2、初始化内存池

实质:还是使用malloc分配内存,不过是以一页4KB分配,当不需要时,再以页为单位进行删除,减少频繁申请/释放的操作

int init_mempool(int block_size, mempool* mp)
{if (!mp) return -1;mp->block_size = block_size;;mp->free_count = MEM_PAGE_SIZE / block_size;mp->mem_ptr = (char*)malloc(MEM_PAGE_SIZE);if (!mp->mem_ptr) return -2;memset(mp->mem_ptr, 0, MEM_PAGE_SIZE);mp->free_list = mp->mem_ptr;//初始化空闲内存块链表int i;char* p = mp->mem_ptr;for (i = 0; i < mp->free_count; i++) {*(char**)(p) = p + block_size;              //节约空间,使用二级指针来存储下一个块的首地址。p += block_size;}*(char**)(p) = NULL;return 0;
}
3、内存池分配内存
void* mempool_alloc(mempool* mp)
{if (!mp || mp->free_count == 0) return NULL;void* p = mp->free_list;if (p) {mp->free_list = *(char**)(p);mp->free_count--;}return p;
}
4、结果

使用二级指针来存储下一个块的首地址。
在这里插入图片描述
在这里插入图片描述

三、总结:

  1. 在确保代码逻辑正确的前提下,使用内存池能显著降低程序长期运行时出现coredump的概率。
  2. 内存池的工作原理是预先申请一大块内存,然后重复利用该内存资源,避免了频繁申请和释放内存带来的开销。
  3. 内存池的分配策略主要分为定长分配和不定长分配两种类型。

四、拓展

可以使用内存管理组件,jemalloc,tcmalloc等。好处就是易上手,缺点就是不一定适用于所有场景。

指标jemalloctcmalloc
线程缓存线程私有tcache,减少锁竞争线程本地缓存
小对象分配
大对象分配
适用场景高并发,多线程应用,高度内存利用率高的场景小块内存分配频繁

五、问题

1、有没有内存池有什么区别?

内存池的使用差异主要体现在程序运行周期上。对于偶尔运行的程序,内存池的作用不大;但对于需要长期稳定运行的服务器应用(7×24小时不间断运行),内存池则能显著提升性能和管理效率。如果程序只运行一两次,那么有没有内存池都行,但是如果程序长期运行,比如服务器,需要长期7*24小时运行,那么有没有内存池的区别就很大了。

代码:
Code
0voice·Github


http://www.hkcw.cn/article/iCgLSETdxB.shtml

相关文章

【TTS】基于GRPO的流匹配文本到语音改进:F5R-TTS

论文地址&#xff1a;https://arxiv.org/abs/2504.02407v3 摘要 我们提出了F5R-TTS&#xff0c;这是一种新颖的文本到语音(TTS)系统&#xff0c;它将群体相对策略优化(GRPO)集成到基于流匹配的架构中。 通过将流匹配TTS的确定性输出重新表述为概率高斯分布&#xff0c;我们的方…

现代密码学入门 | 现代密码学核心特点介绍

在当今互联互通的世界中&#xff0c;数字数据在全球范围内不断流动&#xff0c;安全通信和数据保护的需求从未如此迫切。现代密码学作为数字防御的先锋&#xff0c;提供了一系列复杂的技术和算法&#xff0c;以保护信息免受窥探和恶意行为的侵害。 现代密码学是从其古典前身—…

基于原生JavaScript前端和 Flask 后端的Todo 应用

Demo地址&#xff1a;https://gitcode.com/rmbnetlife/todo-app-js-flask.git Python Todo 应用 这是一个使用Python Flask框架开发的简单待办事项(Todo)应用&#xff0c;采用前后端分离架构。本项目实现了待办事项的添加、删除、状态切换等基本功能&#xff0c;并提供了直观…

【Linux 学习计划】-- 命令行参数 | 环境变量

目录 命令行参数 环境变量 环境变量的本质是什么&#xff1f; 相关配置文件 修改环境变量的相关操作 代码获取env —— environ 内建命令 结语 命令行参数 试想一下&#xff0c;我们的main函数&#xff0c;也是一个函数&#xff0c;那么我们的main函数有没有参数呢&am…

尚硅谷redis7 90-92 redis集群分片之集群扩容

90 redis集群分片之集群扩容 三主三从不够用了&#xff0c;进行扩容变为4主4从 问题&#xff1a;1.新建两个redis实例&#xff0c;怎么加入原有集群&#xff1f;2.原有的槽位分3段&#xff0c;又加进来一个槽位怎么算&#xff1f; 新建6387、6388两个服务实例配置文件新建后启…

Proteus寻找元器件(常见)

一 元件库 二 找元件 1 主控 32 51 输入 stm32 AT89c51 2 找屏幕 oled 3 找按键button 4 电阻、电容 res cap 5 电机驱动 l298n 6 电机 motor 7 滑动变阻器 pot 8 找电源和 GND 9 找晶振 选择 D 开头的 CRYSTAL 10 网络标签

修改Cinnamon主题

~/.themes/Brunnera-Dark/cinnamon/cinnamon.css 1.修改 Tooltip 圆角大小&#xff0c;边框颜色&#xff0c;背景透明度 #Tooltip { border-radius: 10px; color: rgba(255, 255, 255, 0.8); border: 1px solid rgba(255, 255, 255, 0.6); background-color: rgba(0,…

从一到无穷大 #46:探讨时序数据库Deduplicate与Compaction的设计权衡

本作品采用知识共享署名-非商业性使用-相同方式共享 4.0 国际许可协议进行许可。 本作品 (李兆龙 博文, 由 李兆龙 创作)&#xff0c;由 李兆龙 确认&#xff0c;转载请注明版权。 文章目录 引言Compaction AlgorithmsCompact Execution Flow Based On VeloxLocalMergeSource的…

突破DIFY沙箱限制,高效处理大文件

DIFY获取传入文件路径并处理文件内容 由于dify代码沙箱自身的安全限制&#xff0c;用户在沙箱环境下的代码无法实现对系统文件的写入和读取操作。如果想利用dify来处理文件数据&#xff0c;就不得不使用官方提供的文档提取器插件&#xff0c;但是使用该插件提取如.xlsx,.csv等…

比较云计算的四种部署模式:哪个是最佳选择?

在数字化转型浪潮中&#xff0c;企业面临的关键决策之一是如何选择云计算部署模式。公有云、私有云、社区云和混合云并非简单的技术选项&#xff0c;而是关乎业务架构的战略选择。每种模式都代表着不同的资源控制程度、成本结构和安全边界&#xff0c;理解其本质差异是制定有效…

云计算Linux Rocky day02(安装Linux系统、设备表示方式、Linux基本操作)

云计算Linux Rocky day02&#xff08;安装Linux系统、设备表示方式、Linux基本操作&#xff09; 目录 云计算Linux Rocky day02&#xff08;安装Linux系统、设备表示方式、Linux基本操作&#xff09;1、虚拟机VMware安装Rocky2、Linux命令行3、Linux Rocky修改字体大小和背景颜…

项目管理工具Maven

Maven的概念 什么是Maven 什么是依赖管理 对第三方依赖包的管理&#xff0c;可以连接互联网下载项目所需第三方jar包。 对自己开发的模块的管理&#xff0c;可以像引用第三方依赖包一样引用自己项目的依赖包。 什么是项目构建 一、项目构建的定义 项目构建是将源代码经过编…

使用原生前端技术封装一个组件

封装导航栏 navbar-template.html <header><nav><ul><li><a href"index.html"><i class"fas fa-home"></i> 主页</a></li><li><a href"#"><i class"fas fa-theate…

mac mini m4命令行管理员密码设置

附上系统版本图 初次使用命令行管理员&#xff0c;让输入密码&#xff0c;无论是输入登录密码还是账号密码&#xff0c;都是错的&#xff0c;百思不得其解&#xff0c;去网上搜说就是登录密码啊 直到后来看到了苹果官方的文档 https://support.apple.com/zh-cn/102367 https…

使用Vditor将Markdown文档渲染成网页(Vite+JS+Vditor)

1. 引言 编写Markdown文档现在可以说是程序员的必备技能了&#xff0c;因为Markdown很好地实现了内容与排版分离&#xff0c;可以让程序员更专注于内容的创作。现在很多技术文档&#xff0c;博客发布甚至AI文字输出的内容都是以Markdown格式的形式输出的。那么&#xff0c;Mar…

黑马k8s(十七)

一&#xff1a;高级存储 1.高级存储-pv和pvc介绍 2.高级存储-pv 3.高级存储-pvc 最后一个改成5gi pvc3是没有来绑定成功的 pv3没有绑定 删除pod、和pvc&#xff0c;观察状态&#xff1a; 4.高级存储-pc和pvc的生命周期 二&#xff1a;配置存储 1.配置存储-ConfigMap 2.配…

【ABAP 基本数据类型】

ABAP 基本数据类型 一、数值类型 1.1 整数类型 类型关键字长度值范围示例代码标准整型I4字节-2,147,483,648 到 2,147,483,647DATA lv_int TYPE i VALUE 100.短整型INT22字节-32,768 到 32,767DATA lv_short TYPE int2 VALUE -500.无符号整型INT11字节0 到 255DATA lv_flag T…

LearnOpenGL-笔记-其十一

Normal Mapping 又到了介绍法线贴图的地方&#xff0c;我感觉我已经写了很多遍了... 法线贴图用最简单的话来介绍的话&#xff0c;就是通过修改贴图对应物体表面的法线来修改光照效果&#xff0c;从而在不修改物体实际几何形状的前提下实现不同于物体几何形状的视觉效果。 因…

Scratch节日 | 粽子收集

端午节怎么过&#xff1f;当然是收粽子啦&#xff01;这款 粽子收集 小游戏&#xff0c;让你一秒沉浸节日氛围&#xff0c;轻松收集粽子&#xff0c;收获满满快乐&#xff01; &#x1f3ae; 玩法介绍f 开始游戏&#xff1a;点击开始按钮&#xff0c;游戏正式开始&#xff01;…

Linux(9)——进程(控制篇——下)

三、进程等待 1&#xff09;进程等待的必要性 之前提过子进程退出&#xff0c;父进程如果不读取子进程的退出信息&#xff0c;就可能造成“僵尸进程”的问题&#xff0c;从而造成内存泄漏的问题。再者&#xff0c;一旦子进程进入了僵尸状态&#xff0c;那就连kill -9都杀不亖…