一.在Linux操作系统的内存中共有五块空间:代码段,数据段,BBS段,堆和栈
代码段:代码段是用来存放可执行文件的操作指令,也就是说是它是可执行程序在内存种的镜像。代码段需要防止在运行时被非法修改,所以只准许读取操作,而不允许写入(修改)操作——它是不可写的。
数据段:数据段用来存放可执行文件中已初始化全局变量,换句话说就是存放程序静态分配[1]的变量和全局变量。
BSS段[2]:BSS段包含了程序中未初始化全局变量,在内存中 bss段全部置零。
堆(heap):堆是用于存放进程运行中被动态分配的内存段,它大小并不固定,可动态扩张或缩减。当进程调用malloc等函数分配内存时,新分配的内存就被动态添加到堆上(堆被扩张);当利用free等函数释放内存时,被释放的内存从堆中被剔除(堆被缩减)
栈:栈是用户存放程序临时创建的局部变量,也就是说我们函数括弧“{}”中定义的变量(但不包括static声明的变量,static意味这在数据段中存放变量)。除此以外在函数被调用时,其参数也会被压入发起调用的进程栈中,并且待到调用结束后,函数的返回值也回被存放回栈中。由于栈的先进先出特点,所以栈特别方便用来保存/恢复调用现场。从这个意义上将我们可以把堆栈看成一个临时数据寄存、交换的内存区。
二. 下老师布置的程序:
int bss_var;
int data_var0 = 0;
int main(int argc,char **argy){
printf("below are addresses of types's menory/n");
printf("Text location:/n");
printf("/tAddress of main(Code Segement):%p/n",main);//fuction name is a pointer
int stack_var0 = 2;
printf("Stack Location:/n");
printf("/tInitial of stack:%p/n",&stack_var0);
int stack_var1 = 3;
printf("/tnew end of stack:%p/n",&stack_var1);
printf("Data Location:/n");
printf("/tAddress of data_var(Data Segement):%p/n",&data_var0);
static int data_var1 = 4;
printf("/tNew address of data_var(Data Segement):%p/n",&data_var1);
printf("BSS Locatin/n");
printf("/tAddress of bss_var:%p/n",&bss_var);
char *b = sbrk((ptrdiff_t)0);
printf("Heap Location :/n");
printf("/tInitial end of heap:%p/n",b);
brk(b+4);
b = sbrk((ptrdiff_t)0);
printf("/tNew end of heap:%p/n",b);
return 0;
}
运行结果:
ps:
1. sbrk() 和 brk()函数讲解:
int brk(void *end_data_segment);
void *sbrk(intptr_t increment);
brk() 将进程地址空间的data段尾(既内存程序映像的堆尾)设置为end_data_segment所指向的位置.
若成功, 返回0, 否则返回-1.
sbrk()使用差量来调整进程地址空间data段尾的位置, 并返回之前data段尾的地址.
2. ptdiff_t介绍
两个void*指针的差的类型,用于表示两个地址之前的距离,一般定义为int型,可正可负
程序当中
便是取Heap段所在的地址
便是将Heap段向后去四个地址
网上看到的一下前辈的文章,挺不错的。
http://liulong1980.spaces.live.com/blog/