数据段、代码段、堆栈段、BSS段

时间:2023-03-09 15:37:01
数据段、代码段、堆栈段、BSS段

在linux中,进程在内存中一般会分为5个段,用来存放从磁盘载入的程序代码,等.

这五个段分别是:

BSS段:

通常用来存放程序中未初始化的全局变量的一块内存区域.属于静态内存分配.

问题:全局变量不都是默认被初始化为0吗?

(data)数据段:

通常是指用来存放程序中已初始化的全局变量的一块内存区域.数据段属于静态内存分配.

(text)代码段:

通常指用来存放程序执行代码的一块内存区域.这部分区域在程序运行前就已经确定,并且内存区域通常属于只读.(某些架构也允许代码段可写),在代码段中也有可能存在一些只读常量,如只读的字符串等.

(heap)堆:

堆是用来存放进程运行中被动态分配的内存段,它的大小不固定,可动态扩张或缩减.当进程调用malloc分配内存时,就是从堆中分配,当free内存时,堆中去除此内存

(stack)栈:

用户存放程序临时创建的局部变量,也叫自动变量.此外,函数被调用时,参数也被压入发起调用的进程栈中,调用结束后,函数返回值也被存放回栈中.由于栈先进先出特点,所以栈适合用来保存/恢复调用现场.即我们可以把栈看成一个寄存,交换临时数据的内存区.

实例分析

#include <stdio.h>
#include <stdlib.h>

int outa=5;
int outb;
struct try{
int haha;
char heihei;
};
struct try *try_1=NULL;

int main(int argc, char *argv[])
{
        char *a="haha12345678";
        int b=0xa5;
        static int in_s_a=567;
        try_1=(struct try*)malloc(sizeof(struct try));

printf("\taddress_in_a=0x%x,\n  \
                \r\taddress_a=0x%x,\n   \
                \r\taddress_b=0x%x,\n   \
                \r\taddress_outa=0x%x,\n\
                \r\taddress_outb=0x%x,\n\
                \r\taddress_malloc_try_1=0x%x,\n\
                \r\taddress_main=0x%x,\n",
                &in_s_a,&a,&b,&outa,&outb,&try_1,&main);
        return 0;
}

编译:

gcc hello.c

运行:

./a.out

结果:

./a.out

address_in_a=0x60102c,
address_a=0x1b77fe50,
address_b=0x1b77fe5c,
address_outa=0x601028,
address_outb=0x601048,
address_malloc_try_1=0x601040,
address_main=0x400544,

说明:

堆:存放全局变量且在内存底部;

栈:存放局部变量在内存顶部;

代码段:在栈下面

分配内存:在堆中.

BSS段:暂时不明白.

补充:

.bss

未初始化的全局变量和局部静态变量一般放在bss段中.

我们知道未初始化的全局变量和局部静态变量默认值都为0,本来他们也可以被放到.data段中.但是,未初始化的全局变量和局部静态变量默认值都为0,所以为它们在.data段分配空间并存放0没有必要,程序运行的时候它们的确是要内存空间.并且可执行文件必须记录所有未初始化的全局变量和局部静态变量的大小总和,记为.bss段.所以bss段只是为未初始化的全局变量和局部静态变量预留位置而已,它并没有内容,所以它在文件中也不占据空间.