background image

C/C++堆、栈及静态数据区详解

五大内存分区

在 C++中,内存分成 5 个区,他们分别是堆、栈、自由存储区、全局/静态存储
区和常量存储区。

栈,就是那些由编译器在需要的时候分配,在不需要的时候自动清楚的变量
的存储区。里面的变量通常是局部变量、函数参数等。

堆,就是那些由 new 分配的内存块,他们的释放编译器不去管,由我们的应
用程序去控制,一般一个 new 就要对应一个 delete。如果程序员没有释放掉,
那么在程序结束后,操作系统会自动回收。

自由存储区,就是那些由 malloc 等分配的内存块,他和堆是十分相似的,不
过它是用 free 来结束自己的生命的。

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

常量存储区,这是一块比较特殊的存储区,他们里面存放的是常量,不允许
修改(当然,你要通过非正当手段也可以修改,而且方法很多)

明确区分堆与栈

在 bbs 上,堆与栈的区分问题,似乎是一个永恒的话题,由此可见,初学者
对此往往是混淆不清的,所以我决定拿他第一个开刀。

首先,我们举一个例子:

void f() { int* p=new int[5]; }

这条短短的一句话就包含了堆与栈,看到 new,我们首先就应该想到,我们
分配了一块堆内存,那么指针 p 呢?他分配的是一块栈内存,所以这句话的
意思就是:在栈内存中存放了一个指向一块堆内存的指针 p。在程序会先确定
在堆中分配内存的大小,然后调用 operator new 分配内存,然后返回这块内
存的首地址,放入栈中,他在 VC6 下的汇编代码如下:

00401028 push 14h

0040102A call operator new (00401060)

0040102F add esp,4