由于历史原因,C程序一直由下列几部分组成:
(1)
正文段。由
CPU执行的机器指令组成。通常,正文段是可共享的,所以即使是经常执行的程序(如文本编辑程序、C编译程序、shell等)在存储器中也只需有一个副本,另外,正文段常常是制度的,以防止程序由于意外而修改其自身的指令。
(2) 初始化数据段(
数据段)。包含了程序中需赋初值的变量。如外部变量声明(如 int maxcount = 99;),该变量以初值存放在初始化数据段中。
(3) 非初始化数据段(
bss段)。bss这一名称来源于早期汇编程序的一个操作符,意思是“block started by symbo (由符号开始的块)”,在程序开始执行之前,内核将此段初始化为0。例如外部变量声明(long sum[1000];)。
(4)
栈。 自动变量以及每次函数调用时所需保存的信息都存放在此段中。每次函数调用时,其返回地址、以及调用者的环境信息(例如某些机器寄存器)都存放在栈中。然后,新被调用的函数在栈上为其自动和临时变量分配存储空间。通过以这种方式使用栈,C函数可以递归调用。
(5)
堆。 通常在堆中进行动态存储分配。由于历史上形成的惯例,堆位于非初始化数据段顶和栈底之间。
图1 典型的存储器安排
图1显示了程序的逻辑布局——具体实现不一定以这种方式安排其存储空间。
对于VAX上的4.3+BSD,正文段从0位置开始,栈顶则在0x7fffffff之下开始。在VAX机器上,堆顶和栈底之间未用的虚地址空间很大。
从图1还可注意到
未初始化数据段的内容并不存放在磁盘程序文件中。需要存放在磁盘程序文件中的段只有正文段和初始化数据段。
size命令可显示正文段、数据段和bss段的长度。例如: