首先我们必须清楚,地址空间不是物理内存。下图是地址空间的区域分布图
我们先来介绍各区域:
代码区就是保存代码的区域。
静态全局区:保存自动全局变量和static变量(包括static全局变量和static局部变量)。
堆区:由malloc系列函数或new操作符分配的内存。
栈区:保存局部变量
共享区:堆区和栈区的扩展区域。
我们可以通过以下代码来验证这些区域的在地址空间的分布
#include<stdio.h>
int g_val = 0;
int g_arr[10];
int main()
{
static s = 0;
int* p = (int*)malloc(10 * sizeof(int));
int* q = (int*)malloc(10 * sizeof(int));
printf("code:%p\n",main);
printf("static addr:%p\n", &g_val);
printf("static addr:%p\n", &s);
printf("static addr:%p-%p\n", &g_arr[0],&g_arr[9]);
printf("heap addr:%p,%p\n", p,q);
printf("stack addr:%p,%p\n", &p,&q);
free(p);
free(q);
return 0;
}
可以看到地址值是逐渐增大的,所以前面说法无误。
内存分配方式(3种):
(1),从静态存储区域分配
内存在程序编译的时候已经分配好,这块内存在程序的整个运行过程中都存在。包括全局变量,static变量;
(2),在栈上创建
在执行函数时,函数内局部变量的存储单元都可以在栈上创建,
函数执行结束时这些存储单元自动被释放。
栈内存分配运算内置于处理器的指令集中,
效率很高,但是分配的内存容量有限。
(3),从堆上分配(也称为动态内存分布)
程序在运行的时候用malloc或new申请任意多少的内存,
程序员来负责用free或delete释放内存
动态内存的生存期由程序员决定,如果用完之后没有及时free或delete,这块内存就无法释放,出现内存泄漏。
频繁的分配和释放不同大小的堆空间将会产生堆内存碎块。