- 堆区(heap):用于动态内存分配。堆在内存中位于bss区和栈区之间。一般由程序员分配和释放,若程序员不释放,程序结束时有可能由OS 回收。
- 栈区(stack):由编译器自动分配释放,存放函数的参数值、局部变量的值等。其操作方式类似于数据结构中的栈。每当一个函数被调用,该函数返回地址和一些关于调用的信息,比如某些寄存器的内容,被存储到栈区。然后这个被调用的函数再为它的自动变量和临时变量在栈区上分配空间,这就是C实现函数递归调用的方法。每执行一次递归函数调用,一个新的栈框架就会被使用,这样这个新实例栈里的变量就不会和该函数的另一个实例栈里面的变量混淆。
- 全局区(静态区static):存放全局变量、静态数据、常量。程序结束后由系统释放。全局区分为已初始化全局区(data)和未初始化全局区(bss)。
- 常量区(字面量区):存放常量字符串,程序结束后由系统释放
- 代码区(text segment):存放函数题的二进制代码,代码区指令根据程序设计流程依次执行,对于顺序指令,则只会执行一次(每个进程),如果反复,则需要使用跳转指令,如果进行递归,则需要借助栈来实现。代码区的指令中包括操作码和要操作的对象(或对象地址引用)。如果是立即数(即具体的数值,如5),将直接包含在代码中;如果是局部数据,将在栈区分配空间,然后引用该数据地址;如果是data区和bss区,在代码中同样将引用该数据地址。
内存在程序编译阶段就完成好分配,例如static变量,constexpr变量等
在函数执行过程中,由函数体内部创建的变量,此时,变量的创建在栈空间上。栈空间内存分配指令非常高效,由处理器内置指令完成,但是空间大小容量有限,一般为4M,具体和每个计算机的配置有关。
栈空间的内存释放是由函数体的作用域来决定,例如return语句、{ } 结束
堆是从低地址向高地址扩展的数据结构,是不连续的。在堆上分配的内存为动态分配,也即使用malloc/new申请的内存。在堆上的内存需要程序员自己使用malloc/new申请,同时,内存的释放也由程序员使用free/delete进行释放。
堆区可以开辟很大内存,大小受限于计算机的虚拟内存,所以堆获得的空间比较灵活,也比较大,但速度会相对栈区慢一些,也容易产生内存泄漏(内存申请了未释放)。
注意:
静态分配:指发生在程序编译和链接阶段
动态分配:指发生在程序运行阶段
下一篇咱们再继续介绍类对象的内存模型