这篇博客主要介绍 linux 环境下,查看内存占用的两种方式:使用 ps,top等命令;查看/proc/[pid]/下的文件。文章简要介绍了命令的使用方法与一些参数意义,同时对/proc/[pid]/下的文件内容进行了一些详细的介绍。文章内容来自google和自我总结,如有不当之处,欢迎批评指正。
查看Linux内存的方法
linux 下面查看内存有多种渠道,比如通过命令 ps ,top,free, pmap 等,或者通过/proc系统。一般情况下,ps,top,pmap,free可以满足要求,如果需要比较详细和精确地知道整机内存或者某个进程内存的使用情况,可以通过/proc 系统。
使用命令
free : 显示系统可用内存以及已经使用的内存的信息
ps: 查看进程信息,静态,即当前状态
top: 查看进程信息,动态
pstree: 查看进程树
pmap: 根据进程ID查看进程信息
ps vs topps命令–提供系统过去信息的一次性快照,也就是说ps命令能够查看刚刚系统的进程信息。top命令反应的是系统进程动态信息,默认10s更新一次。ps和top都是从/proc目录下读取进程的状态信息,内核把当前系统进程的各种有用信息都放在这个伪目录下。
常见ps命令:
ps -aux: 查看系统所有进程
ps -l: 进查看自己的bash相关进程
top 命令详解,请参考
基本命令:按键盘数字“1”,可监控每个逻辑CPU的状况:键盘“b”(打开/关闭加亮效果)键盘“x”(打开/关闭排序列的加亮效果)”shift + >”或”shift +
具体使用方法,可以使用 man [cmd] 查看。
相关参数说明
VSZ & VIRT进程使用的虚拟内存值总量,包括所有代码,数据,共享库已经被swApped out的。VIRT = SWAP + RES。VSZ来自ps命令, VIRT来自top命令,二者均表示进程占用的虚拟内存大小。假如进程申请100m的内存,但实际只使用了10m,那么它会增长100m,而不是实际的使用量
RES & RSS进程当前使用的内存大小,但不包括swap out. RES = CODE +DATA。包含其他进程的共享RES 来自 top 命令, RSS 来自 ps 命令,两者在表示意义上没有区别,都是从 /cat/proc/[pid]/stat 文件中读取的信息。如果申请100m的内存,实际使用10m,它只增长10m,与VIRT相反关于库占用内存的情况,它只统计加载的库文件所占内存大小
CODE
可执行代码占用的物理内存大小
DATA物理内存中存放数据的大小,在程序运行中需要用到如果top命令没有显示, 按f键显示
SHR共享内存大小除了自身进程的共享内存,也包含其他进程的共享内存计算某个进程所占用物理内存的大小: RES - SHRswap out后,该值会下降。
查看一个进程的内存信息步骤
1.获取进程PID$ ps -aux | grep /usr/sbin/NetworkManagerroot 845 0.0 0.0 387084 13332 ? Ssl 3月28 0:00 /usr/sbin/NetworkManager --no-daemon
2.查看进程的所有线程$ ps mp 845 -o THREAD,tidUSER %CPU PRI SCNT WCHAN USER SYSTEM TIDroot 0.0 - - - - - -root 0.0 19 - - - - 845root 0.0 19 - - - - 1025root 0.0 19 - - - - 1027
3.查看所有子进程$ pstree -p 845NetworkManager(845)─┬─dhclient(30278)├─DNSmasq(1123)├─{gdbus}(1027)└─{gmain}(1025)
使用 /proc 下文件
/proc/[pid]/ 下面与进程内存相关的文件主要有maps , smaps, status。
maps: 文件可以查看某个进程的代码段、栈区、堆区、动态库、内核区对应的虚拟地址
smaps: 显示每个分区更详细的内存占用数据
status: 包含了所有CPU活跃的信息,该文件中的所有值都是从系统启动开始累计到当前时刻
有名与匿名:
一个文件可以映射到进程的一段内存区域中,映射的文件描述符保存在vm_area_struct->vm_file域中,这种内存区域叫做有名内存区域,相反,属于匿名映射内存区域。
maps 文件分析
Proc/[pid]/maps 显示进程映射了的内存区域和访问权限。对应内核中的操作集为 proc_pid_maps_op,具体的导出函数为 show_map 。内核中进程的一段地址空间用一个vm_area_struct结构体表示,所有地址空间存储在task->mm->mmap链表中。
截取一行内容如下:7f4e3f5ca000-7f4e3f674000 r-xp 00000000 08:02 525202 /usr/lib/x86_64-linux-gnu/NetworkManager/libnm-device-plugin-wifi.so
Vm_area_struct每项对应解析如下表所示:
maps文件只能显示简单的分区,smap文件可以显示每个分区的更详细的内存占用数据。
smap 文件分析
截取一段文件,各字段解析如下:7f148b2fa000-7f148b2fb000 rw-p 00026000 08:02 2883675 /lib/x86_64-linux-gnu/ld-2.23.soSize: 4 kB 虚拟内存大小Rss: 4 kB 实际使用物理内存大小 RSS = Shared_Clean+Shared_Dirty+Private_Clean+Private_DirtyPss: 4 kB RSS中私有的内存页面Shared_Clean: 0 kB RSS中共享内存,没有被改写的页面Shared_Dirty: 0 kB RSS中共享内存,被改写的页面Private_Clean: 0 kB RSS中私有内存,未被改写Private_Dirty: 4 kB RSS中私有内存,被改写Referenced: 4 kBAnonymous: 4 kBAnonHugePages: 0 kBShared_Hugetlb: 0 kBPrivate_Hugetlb: 0 kBSwap: 0 kB 处于交换区的页面大小SwapPss: 0 kBKernelPageSize: 4 kB 操作系统一个页面大小MMUPageSize: 4 kB 体系结构MMU一个页面大小Locked: 0 kBVmFlags: rd wr mr mw me dw ac sd
Dirty页面如果没有交换机制的情况下,应该是不能回收的。
分析脚本: