进程的地址空间:TEXT,DATA,BSS,HEAP,STACK

来源:互联网 发布:剑三成男脸型数据网盘 编辑:程序博客网 时间:2024/04/30 12:17

现代操作系统对每个进程都分配了完整的虚拟内存地址空间。进程会把整个地址空间分成多个区间来使用。 程序员最为熟悉的两个区间莫过于。然而还有其他的内存区间来存储代码、静态、全局变量等等。 本文来总结一下这些内存区间到底存的是哪些东西。先看图:

@2x 图片来源: SWAT Blog

虚拟内存技术使得每个进程都可以独占整个内存空间,地址从零开始,直到内存上限。 每个进程都将这部分空间(从低地址到高地址)分为六个部分:

  1. TEXT段:整个程序的代码,以及所有的常量。这部分内存是是固定大小的,只读的。
  2. DATA段,又称GVAR:初始化为非零值的全局变量。
  3. BSS段:初始化为0或未初始化的全局变量和静态变量。
  4. HEAP(堆):动态内存区域,使用mallocnew申请的内存。
  5. 未使用的内存。
  6. STACK(栈):局部变量、参数、返回值都存在这里,函数调用开始会参数入栈、局部变量入栈;调用结束依次出栈。

其中堆和栈的大小是可变的,堆从下往上生长,栈从上往下生长。

由于常量存储在TEXT段中,所有对常量的赋值都将产生segment fault异常。

可以认为BSS段中的所有字节都是0。因为未初始化的全局变量、静态变量都在BSS段中, 所以它们都会被初始化为0,同时类的成员变量也会被初始化为0,但编译器不保证局部变量的初始化。

上面说栈(STACK)是从上到下(高地址到低地址)分配的,而且我们知道, 函数的局部变量的空间是在进入函数体后才分配的,在栈空间里。来个例子来看看吧!

int main(){    char a=0, b=0;    int *p = (int*)&b;    *p = 258;    printf("%d %d", a, b);    return 0;}

输出是

1 2

正常来讲先分配a的地址,再分配b的地址,大小均为1字节,同时它们的地址是连续的! 然后对b的地址进行赋值258(二进制表示是1 0000 0002)。 后面8位0000 0002赋值给了b,前面的1赋值给了a

上述结果和CPU端模式也有关系!我的CPU是小端字节序的~ 低位存的是低字节。

0 0
原创粉丝点击