C程序的存储空间是如何分配?
C语言的内存布局规律
根据内存地址从高到低分别划分为:
- 栈(Stack)
- 未使用的内存空间
- 堆(Heap)
- BSS段(Uninitialized data segment)
- 数据段(Initialized data segment)
- 代码段(Text segment)
代码段
- 代码段(Text segment)通常是指用来存放程序执行代码的一块内存区域。这部分区域的大小在程序运行前就已经确定,并且内存区域通常属于只读。在代码段中,也有可能包含一些只读的常数变量,例如字符串常量等。 例如func str1 str2
(初始化)数据段
-
数据段(Initialized data segment)通常用来存放已经初始化的全局变量和局部静态变量。
-
局部静态变量的生命期是到程序执行结束才消失和全局变量一样,但是作用范围不一样
BSS段(非初始化数据段)
- BSS段(Bss segment/Uninitialized data segment)通常是指用来存放程序中未初始化的全局变量的一块内存区域。BSS是英文Block Started by Symbol的简称,这个区段中的数据在程序运行前将被自动初始化为数字0。
堆heap
- 堆是用于存放进程运行中被动态分配的内存段,它的大小并不固定,可动态扩展或缩小。当进程调用malloc等函数分配内存时,新分配的内存就被动态添加到堆上;当利用free等函数释放内存时,被释放的内存从堆中被剔除。
栈stack
-
大家平时可能经常听到堆栈这个词,一般指的就是这个栈。栈是函数执行的内存区域,通常和堆共享同一片区域。
-
栈内存会自动释放,可以联想到和函数有关的: 存放局部变量,函数的参数,函数的返回值
堆和栈的区别
-
申请方式:
- 堆由程序员手动申请
- 栈由系统自动分配
-
释放方式:
- 堆由程序员手动释放
- 栈由系统自动释放
-
生存周期:
- 堆的生存周期由动态申请到程序员主动释放为止,不同函数之间均可自由访问
- 栈的生存周期由函数调用开始到函数返回时结束,函数之间的局部变量不能互相访问
-
申请之后排序方向:
- 堆和其它区段一样,都是从低地址向高地址发展
- 栈则相反,是由高地址向低地址发展 【栈在最顶部】
注意:
- linux中命令(size 可执行文件名)可以查看所有地址划分情况