最近有人问,我的系统swap快被耗尽了,可是却不知道哪些程序占用了swap,到处发帖问什么命令可以查看。其实linux系统目前并没有这样的命令(或许是有,但我孤陋寡闻不清楚),但是我们可以通过编写脚本来自己找出占用swap的程序。
那么怎样编写这个脚本呢,首先、我们需要知道去哪查看进程占用资源的情况。每个进程都会在/proc下面生成一个以进程号为名字的目录,里面的各个文件就是其在内存中的映像。
其中smaps这个文件,便记载了每个进程每个数据段占用内存的情况。
Swap: 后面的数字,就是该数据段占用的swap大小,我们只要把这个文件里所有出现的swap加起来,就是这个进程所占用的swap大小了。
OK,下面开始写代码:
点击(此处)折叠或打开
#!/bin/bash
echo -e "PID Swap Proc_Name"
# 拿出/proc目录下所有以数字为名的目录(进程名是数字才是进程,其他如sys,net等存放的是其他信息)
for pid in `ls -l /proc | grep ^d | awk ""{ print $9 }""| grep -v [^0-9]`
do
# 让进程释放swap的方法只有一个:就是重启该进程。或者等其自动释放。
# 如果进程会自动释放,那么我们就不会写脚本来找他了,找他都是因为他没有自动释放。
# 所以我们要列出占用swap并需要重启的进程,但是init这个进程是系统里所有进程的祖先进程
# 重启init进程意味着重启系统,这是万万不可以的,所以就不必检测他了,以免对系统造成影响。
if [ $pid -eq 1 ];then continue;fi # Do not check init process
grep -q "Swap" /proc/$pid/smaps 2>/dev/null
if [ $? -eq 0 ];then # 如果占用了swap
# 占用swap的总大小(单位:KB)
swap=$(grep Swap /proc/$pid/smaps | gawk ""{ sum+=$2;} END{ print sum }"")
# 进程名
proc_name=$(ps aux | grep -w "$pid" | grep -v grep | awk ""{ for(i=11;i<=NF;i++){ printf("%s ",$i); }}"")
if [ $swap -gt 0 ];then # 如果占用了swap则输出其信息
echo -e "$pid ${swap} $proc_name"
fi
fi
done | sort -k2 -n | gawk -F"" "" ""{ # 按占用swap的大小排序,再用awk实现单位转换。
# 如:将1024KB转换成1M。将1048576KB转换成1G,以提高可读性。
pid[NR]=$1;
size[NR]=$2;
name[NR]=$3;
}
END{
for(id=1;id<=length(pid);id++)
{
if(size[id]<1024)
printf("%-10s %15sKB %s",pid[id],size[id],name[id]);
else if(size[id]<1048576)
printf("%-10s %15.2fMB %s",pid[id],size[id]/1024,name[id]);
else
printf("%-10s %15.2fGB %s",pid[id],size[id]/1048576,name[id]);
}
}""
本人在某台已占用swap的服务器执行的结果如下:
图片有删减(为方便阅读)
脚本执行结果一目了然,如果某程序占用了大量的swap,则重启该进程即可。否则时间长了,很可能会因为swap耗尽而导致系统死机。