malloc 内存分配机制:brk 与 mmap

news2025/6/6 5:39:00

 一、malloc的两种内存分配策略

malloc 并非直接的系统调用,而是C标准库封装的内存管理函数。它根据应用程序请求的内存大小,智能地选择两种不同的底层机制向操作系统申请内存:

  1. 小块内存分配 (< 128KB):brk() / sbrk() 系统调用

    • 原理: 操作程序的堆顶指针(program break)。通过 brk() 或 sbrk() 将堆顶向高地址方向移动,扩展堆空间,从而获得一块新的连续内存区域。

    • 释放行为: 调用 free() 释放这块内存时,内存通常不会立即归还给操作系统!释放的内存块会被 malloc 的实现(如 ptmallocglibc 的分配器)放入其维护的内部空闲内存池(bins) 中。后续的 malloc 请求会优先从这些池子中寻找合适大小的空闲块复用,避免了频繁的系统调用。

    • 优点: 对于频繁分配释放的小块内存,复用缓存池效率极高。

    • 缺点: 容易在堆内产生内存碎片。大量小内存的分配释放可能导致堆中存在许多小的、不连续的空闲块,虽然总量够用,但无法满足稍大的连续内存请求(外部碎片)。长期运行的程序可能看到堆的“最高水位”不断升高。

  2. 大块内存分配 (>= 128KB):mmap() 系统调用

    • 原理: 在进程的内存映射区域(位于堆和栈之间,通常用于加载共享库、文件映射等)开辟空间。采用 MAP_ANONYMOUS | MAP_PRIVATE 标志,表示分配一块与文件无关、私有的匿名内存。

    • 释放行为: 调用 free() 释放这块内存时,内存会通过 munmap() 系统调用立即归还给操作系统。这块虚拟地址空间和对应的物理内存会被真正释放。

    • 优点: 大块内存单独管理,释放彻底,避免了大块内存长时间占用堆空间导致的外部碎片问题

    • 缺点: 每次分配和释放都涉及系统调用,开销相对较大;每次分配的虚拟地址在初次访问时都会触发缺页中断(Page Fault)以建立物理映射。

二、为什么不是单一策略?

理解 malloc 混合使用 brk 和 mmap 的动机,是掌握其设计精髓的关键:

  1. 为什么不全部使用 mmap

    • 系统调用开销巨大: 每次 mmap 分配都需要从用户态切换到内核态,执行分配逻辑,再切换回用户态。munmap 释放亦然。对于程序中大量、高频的小内存分配(例如几字节到几KB的对象、字符串、结构体),这种开销极其昂贵,会严重拖慢程序性能。

    • 缺页中断开销: mmap 新分配的页面处于“未映射”状态。当程序第一次读写该内存时,会触发缺页中断,由内核分配物理页框并建立映射。虽然 brk 扩展的新区域首次访问也需缺页中断,但 mmap 每次分配的新区域都独立,几乎必然触发新的中断,而 brk 扩展的区域可能是连续的。

  2. 为什么不全部使用 brk

    • 内存碎片: 想象一下,如果所有内存(包括几百MB的大数组)都从堆里分配。程序频繁地分配、释放不同大小的内存块,尤其是大量小块内存,会导致堆内散布着无数小的、不连续的空闲缝隙。即使总的空闲内存足够,也可能无法找到一块连续的、足够大的空间来满足一次较大的 malloc 请求。这就是外部碎片问题。长期运行的程序可能会因为碎片耗尽可用连续空间而崩溃,即使 free 了很多内存。

    • “膨胀”的堆永不收缩: 使用 brk 分配的堆空间,即使 free 了其中的大部分内存,堆顶通常也不会降低(分配器缓存了这些空闲块)。这导致进程的常驻内存集(RSS) 看起来居高不下,可能影响系统整体内存调度和资源监控。

三、使用malloc的注意事项与陷阱

  • 内存泄漏(Memory Leak): 这是最常见也最严重的问题。分配了内存 (malloc), 但忘记释放 (free),或者释放前丢失了指向该内存的唯一指针。程序不断泄漏,最终耗尽可用内存。务必确保分配与释放成对出现,尤其在分支、循环和错误处理路径中。

  • 内存碎片(Memory Fragmentation): 如前所述,brk 策略容易导致外部碎片,mmap 策略对大内存释放更彻底。对于需要长时间运行、频繁分配释放的程序,碎片管理尤为重要。可以考虑使用对象池、内存池等技术缓解。

  • 野指针(Dangling Pointer):

    int *p = (int *)malloc(10 * sizeof(int));
    free(p);  // p 指向的内存已被释放(可能被系统回收或放入缓存池)
    // p 此时是野指针!
    *p = 42;  // 未定义行为!可能导致崩溃、数据损坏等严重后果。

    关键: free(p) 仅释放 p 指向的内存块,并不会改变指针变量 p 本身的值(它仍然指向那块已释放内存的地址)。 安全做法是释放后立即将指针置为 NULL

    free(p);
    p = NULL; // 拴住野指针,后续操作NULL指针通常能更快暴露错误(如崩溃)

  • 检查分配成功: malloc 在内存不足时会返回 NULL永远不要假设 malloc 一定成功!

    int *ptr = (int *)malloc(large_size);
    if (ptr == NULL) {
        // 处理内存分配失败:记录错误、优雅降级或退出
        perror("malloc failed");
        exit(EXIT_FAILURE);
    }

总结与对比:brk vs mmap

特性brk / sbrk (用于 < 128KB)mmap (用于 >= 128KB)
分配区域堆 (Heap)内存映射区域 (Memory Mapped Region)
释放行为缓存到内存池,不立即归还OSmunmap立即归还OS
主要优点高效 (缓存复用,减少系统调用/缺页)避免大块外部碎片,释放彻底
主要缺点易产生内存碎片,堆可能“膨胀”开销大 (系统调用、缺页中断)
适用场景高频、小块内存分配低频、大块内存分配

malloc采用brk和mmap两种方式结合的策略,是在内存分配效率、内存管理和系统资源利用之间取得的一种平衡。了解这些底层实现逻辑,有助于我们在编写 C 程序时更加合理地使用动态内存分配,避免常见的内存问题,提高程序的性能和稳定性。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

设计模式——中介者设计模式(行为型)

摘要 文章详细介绍了中介者设计模式&#xff0c;这是一种行为型设计模式&#xff0c;通过中介者对象封装多个对象间的交互&#xff0c;降低系统耦合度。文中阐述了其核心角色、优缺点、适用场景&#xff0c;并通过类图、时序图、实现方式、实战示例等多方面进行讲解&#xff0…

MinGW-w64的安装详细步骤(c_c++的编译器gcc、g++的windows版,win10、win11真实可用)

文章目录 1、MinGW的定义2、MinGW的主要组件3、MinGW-w64下载与安装 3.1、下载解压安装地址3.2、MinGW-w64环境变量的设置 4、验证MinGW是否安装成功5、编写一段简单的代码验证下6、总结 1、MinGW的定义 MinGW&#xff08;Minimalist GNU for Windows&#xff09; 是一个用…

LabVIEW磁悬浮轴承传感器故障识别

针对工业高端装备中主动磁悬浮轴承&#xff08;AMB&#xff09;的位移传感器故障检测需求&#xff0c;基于 LabVIEW 平台构建了一套高精度故障识别系统。通过集成品牌硬件与 LabVIEW 的信号处理能力&#xff0c;实现了传感器探头故障的实时监测与精准定位&#xff0c;解决了传统…

多线程1(Thread)

认识线程&#xff08;Thread&#xff09; 在进程中&#xff0c;要创建一个进程和销毁一个进程所消耗的硬件和软件资源是巨大的&#xff0c;因此为了优化上述过程&#xff0c;我们引入了“线程”。 线程是系统调度的基本单位。 1&#xff09;线程和进程的关系 可以认为进程包…

NVIDIA DOCA 3.0:引领AI基础设施革命的引擎简析

引言 在当今快速发展的AI时代,大规模AI模型的训练和部署对数据中心基础设施提出了前所未有的挑战。传统的CPU-centric架构已经难以满足超大规模AI工作负载对性能、效率和安全性的需求。NVIDIA于2025年4月正式发布了DOCA 3.0软件框架,这一创新性平台彻底改变了AI基础设施的设计…

小家电外贸出口新利器:WD8001低成本风扇智能控制方案全解析

低成本单节电池风扇解决方案WD8001 用途 低成本单节电池风扇解决方案WD8001用于小功率风扇供电及控制&#xff0c;具有三个档位调节、自动停机及锁机功能。 基本参数 充电参数&#xff1a;输入5V/500mA&#xff0c;满电4.2V&#xff0c;充电指示灯亮&#xff0c;满电后熄灭…

C++实现汉诺塔游戏用户交互

目录 一、模型调整(一)模型定义(二)模型实现1.电脑自动完成部分2.SDL图形显示2.1拿起放下盘子的函数2.2左右移动手指的函数 二、处理用户输入&#xff0c;进行人机分流三、总结四、源码下载 上篇文章使用C语言实现汉诺塔游戏电脑自动完成的步骤&#xff0c;还没有实现用户交互&…

谷歌地图手机版(Google maps)v11.152.0100安卓版 - 前端工具导航

谷歌地图(Google maps)是由谷歌官方推出的一款手机地图应用。软件功能强大&#xff0c;支持本地搜索查找世界各地的地址、地点和商家&#xff1b;支持在街景视图中查看世界各地的360度全景图&#xff1b;支持查找乘坐火车、公交车和地铁的路线&#xff0c;或者查找步行路线等 …

C++核心编程_关系运算符重载

4.5.5 关系运算符重载 作用&#xff1a;重载关系运算符&#xff0c;可以让两个自定义类型对象进行对比操作 /*#### 4.5.5 关系运算符重载 **作用&#xff1a;**重载关系运算符&#xff0c;可以让两个自定义类型对象进行对比操作 */class Person { public:Person(string name, …

T/CCSA 663-2025《医疗科研云平台技术要求》标准解读与深度分析

参考地址:https://www.doc88.com/p-30280431175529.html 引言 随着医疗信息化建设的深入推进,医疗行业正经历从"业务驱动"向"数据驱动"的转型。在这一背景下,中国通信标准化协会(CCSA)于2025年发布了T/CCSA 663-2025《医疗科研云平台技术要求》标准,并…

win11回收站中出现:查看回收站中是否有以下项: WPS云盘回收站

好久没更新了&#xff0c;首先祝所有大朋友、小朋友六一儿童节快乐&#xff0c;真的希望我们永远都不会长大呀&#xff0c;长大真的好累呀(•_•) 免责声明 笔者先来个免责声明吧&#xff0c;被网上的阴暗面吓到了 若读者参照笔者的这篇文章所执行的操作中途或后续出现的任何…

SCDN如何同时保障网站加速与DDoS防御?

在互联网时代&#xff0c;网站既要面对用户访问量的激增&#xff0c;又要抵御层出不穷的网络攻击&#xff0c;特别是DDoS攻击的威胁。SCDN&#xff08;安全内容分发网络&#xff09;作为融合加速与安全的解决方案&#xff0c;如何实现“加速”与“防御”的双重保障&#xff1f;…

项目前置知识——不定参以及设计模式

1.C语言不定参宏函数 c语言中&#xff0c;printf就是一个不定参函数&#xff0c;在使用不定参宏函数时&#xff0c;我们使用__VA_ARGS__来解析不定参&#xff1a; #include <iostream> #include <cstdarg>#define LOG(fmt/*格式*/, .../*用...表示不定参*/) prin…

04powerbi-度量值-筛选引擎CALCULATE()

1、calculate calculate 的参数分两部分&#xff0c;分别是计算器和筛选器 2、多条件calculater与表筛选 多条件有不列的多条件 相同列的多条件 3、calculatertable &#xff08;表&#xff0c;筛选条件&#xff09;表筛选 与calculate用法一样&#xff0c;可以用创建表&…

chromedriver 下载失败

问题描述 chromedriver 2.46.0 下载失败 淘宝https://registry.npmmirror.com/chromedriver/2.46/chromedriver_win32.zip无法下载 解决方法 找到可下载源 https://cdn.npmmirror.com/binaries/chromedriver/2.46/chromedriver_win32.zip &#xff0c;先将其下载到本地目录(D…

Weather app using Django - Python

我们的任务是使用 Django 创建一个 Weather 应用程序&#xff0c;让用户可以输入城市名称并查看当前天气详细信息&#xff0c;例如温度、湿度和压力。我们将通过设置一个 Django 项目&#xff0c;创建一个视图来从 OpenWeatherMap API 获取数据&#xff0c;并设计一个简单的模板…

机器视觉2,硬件选型

机器视觉1&#xff0c;学习了硬件的基本知识和选型&#xff0c;现在另外的教材巩固知识 选相机 工业相机选型的保姆级教程_哔哩哔哩_bilibili 1.先看精度多少mm&#xff0c;被检测物体长宽多少mm》分辨率&#xff0c; 选出合理范围内的相机 2.靶面尺寸&#xff0c;得出分…

电阻电容的选型

一、电阻选型 1.1安装方式 贴片电阻体积小&#xff0c;适用于SMT生产&#xff1b;功率小&#xff1b;易拆解插件电阻体积大&#xff1b;功率大&#xff1b;不易脱落 1.2阻值 电阻的阻值是离散的&#xff0c;其标称阻值根据精度分为E6、E12、E24、E48、E96、E192六大系列&am…

12.springCloud AlibabaSentinel实现熔断与限流

目录 一、Sentinel简介 1.官网 2.Sentinel 是什么 3.Sentinel 的历史 4.Sentinel 基本概念 资源 规则 5.Sentinel 功能和设计理念 (1).流量控制 什么是流量控制 流量控制设计理念 (2).断降级 什么是熔断降级 熔断降级设计理念 (3).系统自适应保护 6.主要工作机制…

vSOME/IP与ETAS DSOME/IP通信的问题解决方案

✅ 一、服务版本不匹配导致 Handover 问题 —— 需要更新 VSOMEIP 代码逻辑 📌 问题描述: 在 SOME/IP 通信中,发布者(offer)与订阅者(subscribe)之间存在服务版本不一致的问题,导致 Handover(切换)失败。 ✅ 解决方案: 需要在 offer_service 和 subscribe 接口中…