可执行程序包括BSS段、数据段、代码段。
在类UNIX系统下可使用size命令查看可执行文件的段大小信息。如size a.out:
~/Desktop/MyC$ size a.outtext data bss dec hex filename1672 600 8 2280 8e8 a.out
1.数据段存放已初始化的全局变量和静态变量,数据段属于静态内存分配。
2.BSS段(Block Started by Symbol)存放未初始化的全局变量和静态变量。
BSS段的数据是可读写的,链接器从可执行文件中得到BSS段的大小,然后申请得到这块内存空间,这块内存空间紧跟在数据段的后面。由此可知BSS段并不占用可执行文件的大小。在使用BSS段之前BSS段会自动初始化为0。所以,未初始的全局变量和静态变量在程序执行之前已经是0了。BSS段属于静态内存分配。
包含数据段和BSS段的整个区段通常称为数据区。
3.代码段:用来存放程序的代码内存空间。它的大小在程序运行前就已经确定了,并且该区域只能读不能写。在代码段中,也有可能包含了一些只读的常数变量,例如字符串常量等。
代码段和数据段在编译时已经分配了空间,而BSS段则在程序被调入内存后才分配的。因此BSS段(未进行初始化的数据)的内容并不存放在磁盘上的程序文件中。需要存放在程序文件中的只有代码段和数据段(存放已初始化的全局变量和静态变量)的内容。
程序编译后生成的目标文件至少含有这三个段,这三个段的大致结构图如下所示:
可执行程序在运行时会多出两个区域:栈区和堆区。
4.栈区:由操作系统自动分配和释放 ,存放函数的参数值,局部变量的值等。每当一个函数被调用时,该函数的返回类型和一些调用的信息也会被存放到栈中。然后这个被调用的函数再为它的自动变量和临时变量在栈上分配空间。每调用一个函数一个新的栈就会被使用。栈区是从高地址位向低地址位增长的,是一块连续的内存区域,最大容量是由系统预先定义好的,申请的栈空间超过这个界限时会提示溢出,用户能从栈中获取的空间较小。
小知识:
自动变量:什么是自动变量呢?自动变量就是指在函数内部定义使用的变量。只能在函数内部使用它。自动变量是局部变量,即它的作用区域是在定义它的函数内部。由于自动变量在定义它的函数的外部是不可见的,所以允许我们在这个函数外部或者其他的函数内部定义同名的变量。计算机在执行这个函数的时候,创建这个变量并为它分配内存,当函数执行完毕返回后,自动变量就会被销毁。为自动变量分配内存就是压栈,而函数返回时就退栈。
临时变量是指在未在程序开头部分声明的,待使用它时才声明类型的变量。常见的如函数中定义的变量,循环语句、条件语句中声明定义的变量。这些变量可与主程序中的变量同名,在其作用域里,主程序中的同名变量一般无法调用。并且这种变量有效存在时间是从变量声明开始到相应程序段(循环结构或函数体)结束。
5.堆区:用于动态分配内存,位于BSS和栈中间的地址区域。由程序员亲自用malloc()申请分配和用free()释放。堆是从低地址位向高地址位增长,采用链式存储结构。频繁的 malloc/free造成内存空间的不连续,产生碎片。当申请堆空间时库函数是按照一定的算法搜索可用的足够大的空间。因此堆的效率比栈要低的多。
最后提示一下:内存泄漏是指内存分配出去以后,你再也访问不到了。