我们都知道进程运行时,会有一个栈空间(stack)和一个堆空间(heap), 栈空间用于函数调用和局部变量,堆空间是C语言的 malloc 来分配的全局指针。这些都是进程的私有数据,除了这些,还有映射进来的动态库,进程间的共享内存等共享空间。另外,操作系统还支持预留虚拟地址空间的功能(延迟分配),也就是在读写该内存的时候,操作系统才进行物理内存的分配,因此进程占用的空间情况还是比较复杂的。下面简单地说明一下。
Linux VSZ:Virtual Memory Size(虚拟内存大小)。进程占用的全部地址空间,共享库,预分配内存,交换分区等都包含在里面。因此,它远远大于实际的占用的内存空间。
RSS:Resident Set Size(驻留集大小), 实际占用的物理内存,它包含共享库,但不包含在交换分区的空间。随着程序的运行,RSS也会跟着增长,VSZ将是它的上限。
PSS:Proportional Set Size, 也是实际分配的物理内存,与RSS的区别是,它以平分的方式来计算共享库的大小(共享库 / 进程个数), RSS会把共享库的大小全部计算进来。
USS:Unique Set Size, 进程的私有内存(独自使用的库,堆等空间),不包含共享的内存空间。
ANON: Anonymous memory,匿名内存 —— 没有文件关联的内存页面。Linux会自动映射文件到内存,读取的文件后,会自动缓存到内存。如果,应用程序只是使用mmap(MAP_ANONYMOUS) 分配一些内存页面没有文件关联,就称为“匿名内存”。
Dirty: dirty pages , 脏页大小 —— 还没有写回到硬盘的缓存页面。
VIRT: 同VSZ。
RES: 同RSS。
假如进程A(2k),只依赖动态库B(1000k) ;A 分配 128k的匿名空间,200k的堆栈和堆的空间——实际使用100k。其中动态库B被 2个进程共享,实际加载200k,那么 ——
VSZ = 2k + 1000k + 128k + 200k = 1230k
RSS = 2k + 200k + 128k + 100k = 430k
PSS = 2k + 200k / 2 + 128k + 100k = 330k
USS = 2k + 128k + 100k = 230k
ANON = 128k
工具 进程的内存原始数据的主要来源于 proc 目录下的2个文件,可以 直接cat 查看。
-
/proc/{$PID}/smaps
-
/proc/{$PID}/maps
maps 是以地址空间段来输出内容,而 smaps 则以文件,堆,堆栈来组织内容。
1: ps aux
单独查看某个进程
2: top
3: pmap -[x/X/XX] {$PID}
x 越大,越多,信息越详细。
4: smem
Windows
windows 有一个 [reserve/commit] 的概念,VirtualAlloc 可以预先保留一个连续的虚拟内存空间,实际上没有内存分配。当需要使用的时候,再次使用 VirtualAlloc 去提交(commit)需要使用的区域,当这个内存区域被读写的时候,操作系统才进行内存分配。 因此,commit空间一般偏大。
进程正在使用的物理内存称为工作集(working set),分为私有工作集和共享工作集。可执行程序在运行过程中分配的内存,称为私用内存。如果再加上文件cache,dll等共享内存(page file),就是进程使用的全部物理内存——工作集。工作集不包含交换分区,只是当前被分配的物理内存,而私有数据则包含交换分区的内容 —— 假如:有10M的私有数据,8M被换出,那么工作集只把还在内存里的2M大小计算在内。
进程的虚拟内存空间又称为“Commit Size Limit”, 它的大小有两种计算方式:
Vmmap是windows官方提供的一个工具,下图是软件的窗口, WS即 工作集,任务管理器里显示的是Private WS(专用工作集)的值。
Size: 软件向操作系统申请的最大物理内存空间,OS在使用时才分配,因此进程实际占用的空间会比这个值小,但在使用过程中,占用空间可能会一直增加。
Committed: 已提交到OS 的内存空间,读写内存页面时分配,上限是 Size。
Total WS = Private WS + Shareable WS 。
---
欢迎大家来我的公众号交流: 般若程序蝉