进程映像
-
程序是保存在磁盘上的可执行文件。
-
运行程序时,需要将可执行文件加载到内存,形成进程。
-
一个程序(文件)可以同时存在多个进程(内存)。
-
进程在内存空间中的布局就是进程映像,从低地址到高地址依次为:
-
虚拟内存
虚拟内存是计算机系统内存管理的一种技术。它使得应用程序认为它拥有连续的可用的内存(一个连续完整的地址空间),而实际上,它通常是被分隔成多个物理内存碎片,还有部分暂时存储在外部磁盘存储器上,在需要时进行数据交换。
1. 每个进程都有各自互独立的4G字节虚拟地址空间。
2. 用户程序中使用的都是虚拟地址空间中的地址,永远无法直接访问实际物理内存地址。
3. 虚拟内存到物理内存的映射由操作系统动态维护。
4. 虚拟内存一方面保护了操作系统的安全,另一方面允许应用程序,使用比实际物理内存更大的地址
空间。
4G进程地址空间分成两部分:
[0, 3G)为用户空间,
如某栈变量的地址0xbfc7fba0=3,217,554,336,约3G;
[3G, 4G)为内核空间。
对内存的越权访问,或试图访问没有映射到物理内存的虚拟内存,将导致段错误。用户空间对应进程,进程一切换,用户空间即随之变化。
内核空间由操作系统内核管理,不会随进程切换而改变。
进行内存映射,而用户空间的页表则每个进程一份。
每个进程的内存空间完全独立。不同进程之间交换虚拟内存地址是毫无意义的。
malloc底层实现
brk
首先来讲增量方式分配虚拟内存brk和sbrk
void* sbrk ( intptr_t increment );
返回上次调用brk/sbrk后的末尾地址,失败返回-1。
increment取值:内存增量(以字节为单位)
0 - 获取末尾地址。
>0 - 增加内存空间。
<0 - 释放内存空间。
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
int main(){
//如果失败返回 -1(0xffffffff) 内存地址
char *p1 = (char*)sbrk(1);//申请1个字节的内存空间
*p1 = 'A';
printf("%p\n",p1);
void *p2 = sbrk(0);//获取动态内存的末尾位置
printf("%p\n",p2);
int *p3 = (int *)sbrk(4);
printf("%p\n",p3);
*p3 = 1024;
//void *p4 = sbrk(-5);//全部释放了