堆空间与栈空间

@TOC

一、内存管理:栈空间与堆空间

1.栈空间

  • 特点:由编译器自动分配/释放,存放局部变量、函数参数等。
  • 指针定义:指针变量本身存储在栈中(如 int *p;),但其指向的内存可能是堆或栈。
  • 限制:栈空间较小(默认约几MB),超出会导致栈溢出(Stack Overflow)。

2.堆空间

  • 特点:手动分配/释放,容量更大,生命周期由开发者控制。
  • 动态分配函数
void* malloc(size_t size);    // 分配未初始化内存
void* calloc(size_t num, size_t size); // 分配并初始化为0
void* realloc(void* ptr, size_t new_size); // 调整已分配内存大小
void free(void* ptr);         // 释放内存
  • 使用示例
int *p = (int*)malloc(sizeof(int)); // 显式类型转换
if (p == NULL) { /* 处理分配失败 */ }
*p = 10;
free(p);
p = NULL; // 避免悬空指针(Dangling Pointer)

二、动态内存分配详解

1.malloc的注意事项

  • 返回值:返回 void* 类型指针,需显式转换为目标类型。
  • 初始化:分配的内存未初始化,可能含随机值(需手动初始化或使用 calloc)。
  • 失败处理:分配失败返回 NULL,需检查返回值。

2.内存泄露

  • 原因:未释放不再使用的堆内存。
  • 后果:程序长时间运行后内存耗尽,导致崩溃。
  • 避免方法
    • 确保每个malloc对应一个free
    • 使用工具(Valgrind)检测内存泄漏。

3.悬空指针与野指针

  • 悬空指针:指向已释放内存的指针,释放后应立即置NULL
  • 野指针:未初始化的指针,可能指向非法地址,需初始化为NULL

三、函数中的动态内存分配

1.函数内部分配堆空间

  • 允许场景:若需在函数外部使用内存,可在函数内部分配并返回指针。
  • 责任传递调用者必须负责释放内存。
  • 示例
int* create_int(int value) {
    int *p = (int*)malloc(sizeof(int));
    if (p != NULL) *p = value;
    return p;
}
// 调用者需调用 free() 释放

四、取模运算(%)规则

  • C语言规则:余数符号与被除数符号一致。
    • -5%-3=-2(被除数-5为负,余数也为负)
    • 5%-3=2(被除数5为正,余数为正)
堆地址空间和栈空间存在多方面的区别: ### 管理方式 栈由编译器自动管理,无需程序员手工控制;而堆的分配和释放由程序员负责,例如使用`new`分配堆内存,使用`delete`释放堆内存,使用`malloc`分配堆内存,使用`free`释放堆内存[^3][^5]。 ### 空间大小 在Windows下,栈是一块连续的内存区域,栈顶地址和最大容量系统预先规定好,能获得的空间较小;堆是不连续的内存区域,大小受限于计算机系统的有效虚拟内存空间,获得的空间比较灵活且较大,堆空间理论上有几G的空间[^1][^2]。 ### 内存碎片 由于大量`malloc()/free()`或`new/delete`的使用,堆容易造成大量的内存碎片;栈是先进后出的队列,不会产生内存碎片[^3]。 ### 生长方向 栈是向低地址扩展的数据结构,生长方向向下;堆是向高地址扩展的数据结构,生长方向向上,但后申请的内存空间不一定在先申请的内存空间后面,因为堆内存块是动态分配的[^1][^2]。 ### 分配方式 栈可以是静态分配和动态分配,不过栈的动态分配由编译器释放;堆则是动态分配,由程序员手动控制释放[^3]。 ### 缓存级别 栈使用一级缓存,通常在被调用时处于存储空间中,调用完毕立即释放;堆存放在二级缓存中,生命周期由虚拟机的垃圾回收算法决定,调用对象的速度相对较低[^3]。 ### 分配效率 栈是机器系统提供的数据结构,计算机底层对栈提供支持,有专门寄存器存放栈地址,压栈出栈有专门指令,效率较高;堆由C/C++函数库提供,分配时需按一定算法搜索可用空间,可能调用系统功能增加程序数据段内存空间,效率较低[^3]。 以下是一个简单的C++代码示例,展示栈和堆的使用: ```cpp #include <iostream> int main() { // 栈上分配内存 int stackVar = 10; std::cout << "Stack variable value: " << stackVar << std::endl; // 堆上分配内存 int* heapVar = new int(20); std::cout << "Heap variable value: " << *heapVar << std::endl; // 释放堆上的内存 delete heapVar; return 0; } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值