当前位置: 代码迷 >> 综合 >> Linux内存管理(进程映像、虚拟内存)
  详细解决方案

Linux内存管理(进程映像、虚拟内存)

热度:97   发布时间:2024-02-22 18:23:14.0

一、内存管理
用户层:
STL 自动分配/释放内存 调用C++
C++ new/delete 调用C
C malloc/free 调用POSIX
POSIX brk/sbrk 调用Linux
Linux mmap/munmap 调用内核
系统层
kernal kmalloc/vmalloc 调用驱动
driver get_free_page
二、进程映像
程序是存储在磁盘上的可执行文件,当执行程序时,系统会将可执行文件加载到同,内存中形成进程(一个程序可以同加载出多个进程)。
进程的内存空间分布就是进程映像,从低地址到高地址依次是:
text 代码段 二进制指令,常量(字符串字面值,被const修改过的原data段的数据)
只能读,如果修改会产生段错误
data 数据段 初始化过的全局变量和静态变量
bss 静态数据段 未初始化过的全局变量和静态变量
该段内存会被清理为0。
heap 堆 体量比较大的数据,结构变量
手动管理,释放时间可控,空间大,使用时与指针配合
使用麻烦,可能产生内存碎片和内存泄漏。
stack 栈 局部、块变量
大小有限,自动分配释放,不会产生内存碎片、泄漏
environ 环境变量表 环境变量
每个进程一份,修改并不会影响其它进程
argv 命令行参数
程序执行时附加的参数
三、虚拟内存:
1、系统会为每个进程分4G的虚拟内存空间。
32个0 ~ 32个1 地址范围。
2、用户只能使用虚拟地址,无法直接使用物理内存。
3、虚拟地址与物理内存进行映射才能使用,否则就会产生段错误。
4、虚拟地址与物理内存的映射由操作系统动态维护。
5、让用户使用虚拟地址一方面为了安全,另一方面操作系统可以让应用程序使用比实际物理内存更的地址空间。
6、4G的虚拟地址分为两部分
[0~3) 用户空间
[3~4) 内核空间
7、用户空间中的代码不能直接访问内核空间代码和数据,可以通过调用系统API切换到内核态,间接的与内核交换数据。
8、对虚拟内存越界访问(使用没有映射过的内存)将导致段错误。
四、内存管理总结:
1、mmap/munmap 底层不维护任何东西,只返回一个映射后的内存首地址,所映射的内存位于堆中。
2、brk/sbrk底层维护一个指针,记录所映射的内存结尾,所映射的内存也位于堆中,底层调用的是mmap/munmap。
3、malloc/free底层维护一个双向链表和必须的控制信息,所映射的内存也位于堆中,底层调用的是brk/sbrk。
4、每个进程都有4G(32位系统)的虚拟内存空间,虚拟内存只是个数据,必须与物理内存建立映射关系才能使用。
5、平时所说内存的分配与释放有两层含义。
1、权限的分配与释放
2、映射关系的取消与建立
6、重点是理解Linux系统的内存管理机制,而不是brk/sbrk/mmap/munmap的用法。