background image

C++内存管理详解

1. 内存分配方式
1.1 分配方式简介
  在 C++中,内存分成 5 个区,他们分别是堆、栈、自由存储区、全局/静态存储区和常

 

量存储区。
  栈,在执行函数时,函数内局部变量的存储单元都可以在栈上创建,函数执行结束
时这些存储单元自动被释放。栈内存分配运算内置于处理器的指令集中,效率很高,但是
分配的内存容量有限。
  堆,就是那些由 new 分配的内存块,他们的释放编译器不去管,由我们的应用程序
去控制,一般一个 new 就要对应一个 delete。如果程序员没有释放掉,那么在程序结束

 

后,操作系统会自动回收。
  自由存储区,就是那些由 malloc 等分配的内存块,他和堆是十分相似的,不过它是
用 free

 

来结束自己的生命的。

  全局/静态存储区,全局变量和静态变量被分配到同一块内存中,在以前的 C 语言中,
全局变量又分为初始化的和未初始化的,在 C++里面没有这个区分了,他们共同占用同

 

一块内存区。
  常量存储区,这是一块比较特殊的存储区,他们里面存放的是常量,不允许修改。
1.2 明确区分堆与栈

 

  首先,我们举一个例子:
                           void f() { int* p=new int[5]; }  
00401058   push        14h
0040105A   call        operator new (00411580)
0040105F   add         esp,4
00401062   mov         dword ptr [ebp-8],eax
00401065   mov         eax,dword ptr [ebp-8]
00401068   mov         dword ptr [ebp-4],eax
  这条短短的一句话就包含了堆与栈,看到 new,我们首先就应该想到,我们分配了
一块堆内存,那么指针 p

 

呢?他分配的是一块栈内存,所以这句话的意思就 是:在栈内

存中存放了一个指向一块堆内存的指针 p。在程序会先确定在堆中分配内存的大小,然后
调用 operator new 分配内存,然后返回这块内存的首地址,放入栈中,他在 VC6 下的
汇编代码如下:
     00401028 push 14h 
     0040102A call operator new (00401060) 
     0040102F add esp,4 
     00401032 mov dword ptr [ebp-8],eax 
     00401035 mov eax,dword ptr [ebp-8] 
     00401038 mov dword ptr [ebp-4],eax 
  这里,我们为了简单并没有释放内存,那么该怎么去释放呢?是 delete p 么?澳,
错了,应该是 delete []p,这是为了告诉编译器:我删除的是一个数组,VC6 就会根据
相应的 Cookie 信息去进行释放内存的工作。
1.3 堆和栈究竟有什么区别?
  主要的区别由以下几点: