C++ 进阶笔记:栈内存 vs 堆内存

C++ 进阶笔记:栈内存 vs 堆内存

一、 内存的基本概念

在 C++ 程序运行时,RAM(内存)会被划分为多个区域,其中最令开发者关心的两个区域就是 栈 (Stack)堆 (Heap)

  • 物理位置:两者都在 RAM 中,物理本质没有区别。
  • 分配逻辑:区别在于操作系统和编译器如何管理这两块区域。

二、 栈内存 (Stack) ------ 极致的速度

1. 工作原理

栈的工作方式类似于"叠盘子"。它有一个 栈指针 (Stack Pointer),分配内存只需移动该指针。

  • 分配:向下移动指针(在大多数系统上地址由高到低)。
  • 释放:向上移动指针。

2. 特点

  • 生命周期 :由作用域 {} 自动管理。一旦离开大括号,内存立即释放(Pop)。
  • 速度 :极快。在汇编层面通常只需 一个 CPU 指令 (如 sub esp, 4)。
  • 局限性 :空间有限(通常为 1MB - 2MB)。如果分配过大(如 int arr[1000000]),会导致 栈溢出 (Stack Overflow)

三、 堆内存 (Heap) ------ 灵活性与代价

1. 工作原理

堆是一块巨大的空闲存储区。

  • 分配 :使用 new 关键字。系统会查找 空闲列表 (Free List) 来寻找足够大的连续空间。
  • 释放 :必须手动调用 delete(或使用智能指针),否则会导致 内存泄漏

2. 特点

  • 生命周期:由程序员手动控制,可以跨越函数作用域存在。
  • 速度 :较慢。因为 new 涉及到繁琐的"簿记"工作,甚至需要向操作系统申请更多物理内存。
  • 优势:空间几乎仅受物理内存限制,适合存储大数据(如数据库缓冲区、大型纹理)。

四、 汇编与底层对比 (Debug 模式下)

Cherno 在视频中对比了两者在底层的真实开销:

  • 栈分配 (int v = 5)
    直接将值存入对应的内存地址。指令极其简单。
  • 堆分配 (int v = new int(5)) *:
    1. 调用 operator new
    2. 内部调用 malloc
    3. 执行复杂的查找空闲空间、记录分配大小等逻辑。
    4. 最终才返回指针。

五、 核心结论:如何选择?

  1. 默认优先用栈:只要对象不需要在函数结束后存活,且大小不是特别巨大,永远优先选择栈分配。
  2. 必须用堆的情况
    • 对象需要在函数结束后继续存在(需手动控制生命周期)。
    • 分配的数据量非常大(超过几百 KB 或 MB 级别)。
    • 对象的具体大小在编译时无法确定,且可能非常大。

六、 进阶思考

  • 内存池技术 (Memory Pooling) :为了结合"堆的容量"和"栈的速度",数据库通常会在启动时在堆上 new 一大块空间,然后自己在内部模拟栈的分配逻辑(手动移动指针)。这被称为"自定义内存分配器"。
相关推荐
IT 行者3 小时前
Web逆向工程AI工具:JSHook MCP,80+专业工具让Claude变JS逆向大师
开发语言·javascript·ecmascript·逆向
程序员 沐阳4 小时前
JavaScript 内存与引用:深究深浅拷贝、垃圾回收与 WeakMap/WeakSet
开发语言·javascript·ecmascript
Mr_Xuhhh5 小时前
Java泛型进阶:从基础到高级特性完全指南
开发语言·windows·python
He1955015 小时前
wordpress搭建块
开发语言·wordpress·古腾堡·wordpress块
老天文学家了6 小时前
蓝桥杯备战Python
开发语言·python
赫瑞6 小时前
数据结构中的排列组合 —— Java实现
java·开发语言·数据结构
初夏睡觉6 小时前
c++1.3(变量与常量,简单数学运算详解),草稿公放
开发语言·c++
升职佳兴6 小时前
C盘爆满自救:3步无损迁移应用数据到E盘(含回滚)
c语言·开发语言
ID_180079054736 小时前
除了 Python,还有哪些语言可以解析 JSON 数据?
开发语言·python·json
阿拉斯攀登7 小时前
从入门到实战:CMake 与 Android JNI/NDK 开发全解析
android·linux·c++·yolo·cmake