我认为我的LAMP应用程序中可能发生内存泄漏(内存被耗尽,交换开始被使用,等等)。如果我可以看到各个进程使用了多少内存,则可以帮助我解决问题。有没有办法让我在* nix中查看此信息?
我认为我的LAMP应用程序中可能发生内存泄漏(内存被耗尽,交换开始被使用,等等)。如果我可以看到各个进程使用了多少内存,则可以帮助我解决问题。有没有办法让我在* nix中查看此信息?
Answers:
获得正确的内存使用率比人们想象的要难。我能找到的最好方法是:
echo 0 $(awk '/TYPE/ {print "+", $2}' /proc/`pidof PROCESS`/smaps) | bc
其中“ PROCESS”是要检查的过程的名称,而“ TYPE”是以下之一:
Rss
:常驻内存使用情况,该进程使用的所有内存,包括该进程与其他进程共享的所有内存。它不包括掉期;Shared
:该进程与其他进程共享的内存;Private
:此过程使用的私有内存,您可以在此处查找内存泄漏;Swap
:交换进程使用的内存;Pss
:比例设置大小,良好的整体内存指示器。它是为共享而调整的Rss:如果某个进程在其他10个进程之间具有1MiB专用和20MiB共享,则Pss为1 + 20/10 = 3MiB其他有效值是Size
(即虚拟大小,几乎没有意义)和Referenced
(当前标记为引用或访问的内存量)。
您可以使用watch
或其他bash-script-fu来监视要监视的进程的那些值。
有关更多信息smaps
:http : //www.kernel.org/doc/Documentation/filesystems/proc.txt。
echo 0 $(sudo awk '/TYPE/ {print "+", $2}' /proc/PID/smaps) | bc
awk
传递而不是传递给bc
?awk 'BEGIN { used=0 }; /TYPE/ { used += $2 } END { print used }' /proc/PID/smaps
将为您提供KB大小。
我不知道为什么答案看起来如此复杂...这样做似乎很简单ps
:
mem()
{
ps -eo rss,pid,euser,args:100 --sort %mem | grep -v grep | grep -i $@ | awk '{printf $1/1024 "MB"; $1=""; print }'
}
用法示例:
$ mem mysql
0.511719MB 781 root /bin/sh /usr/bin/mysqld_safe
0.511719MB 1124 root logger -t mysqld -p daemon.error
2.53516MB 1123 mysql /usr/sbin/mysqld --basedir=/usr --datadir=/var/lib/mysql --plugin-dir=/usr/lib/mysql/plugin --user=mysql --pid-file=/var/run/mysqld/mysqld.pid --socket=/var/run/mysqld/mysqld.sock --port=3306
.zshrc
rss
用于计算的列(Resident Set Size)包含共享库中的内存,因此将数字丢弃。就我而言,进程使用的内存比系统可用的内存更多。
使用ps
查找应用程序的进程ID,然后使用top -p1010
(将1010替换为实际进程ID)。RES列是已使用的物理内存,VIRT列是已使用的虚拟内存-包括库和交换的内存。
可以使用“ man top”找到更多信息
top -p`ps -ef | grep -i $NAME_OF_PROCESS | grep -v grep | gawk '{print $2}'`
pmap $(pgrep -f -u username /usr/bin/gnome-shell) | sed -n -e 's/ total \+//p' | numfmt --from=iec
1724678144
使用top
或htop
并注意“ RES”(驻留内存大小)列。
谢谢。我用它来创建这个简单的bash脚本,该脚本可用于监视进程及其内存使用情况:
$ watch watchmypid.sh
#!/bin/bash
#
PROCESSNAME=changethistoyourprocessname
MYPID=`pidof $PROCESSNAME`
echo "=======";
echo PID:$MYPID
echo "--------"
Rss=`echo 0 $(cat /proc/$MYPID/smaps | grep Rss | awk '{print $2}' | sed 's#^#+#') | bc;`
Shared=`echo 0 $(cat /proc/$MYPID/smaps | grep Shared | awk '{print $2}' | sed 's#^#+#') | bc;`
Private=`echo 0 $(cat /proc/$MYPID/smaps | grep Private | awk '{print $2}' | sed 's#^#+#') | bc;`
Swap=`echo 0 $(cat /proc/$MYPID/smaps | grep Swap | awk '{print $2}' | sed 's#^#+#') | bc;`
Pss=`echo 0 $(cat /proc/$MYPID/smaps | grep Pss | awk '{print $2}' | sed 's#^#+#') | bc;`
Mem=`echo "$Rss + $Shared + $Private + $Swap + $Pss"|bc -l`
echo "Rss " $Rss
echo "Shared " $Shared
echo "Private " $Private
echo "Swap " $Swap
echo "Pss " $Pss
echo "=================";
echo "Mem " $Mem
echo "=================";
如果您没有当前或长期运行的流程需要跟踪,可以使用/usr/bin/time
。
这与Bash不同time
(如您所见)。
例如
# /usr/bin/time -f "%M" echo
2028
这是“进程在其生存期内的最大驻留集大小,以千字节为单位”(从手册页中引用)。即与RES中的相同top
等人。
您可以从中得到更多/usr/bin/time
。
# /usr/bin/time -v echo
Command being timed: "echo"
User time (seconds): 0.00
System time (seconds): 0.00
Percent of CPU this job got: 0%
Elapsed (wall clock) time (h:mm:ss or m:ss): 0:00.00
Average shared text size (kbytes): 0
Average unshared data size (kbytes): 0
Average stack size (kbytes): 0
Average total size (kbytes): 0
Maximum resident set size (kbytes): 1988
Average resident set size (kbytes): 0
Major (requiring I/O) page faults: 0
Minor (reclaiming a frame) page faults: 77
Voluntary context switches: 1
Involuntary context switches: 0
Swaps: 0
File system inputs: 0
File system outputs: 0
Socket messages sent: 0
Socket messages received: 0
Signals delivered: 0
Page size (bytes): 4096
Exit status: 0
gnu-time
软件包提供了正确的实用程序。它安装了一个名为的实用程序gtime
,它可以满足您的需求。
您可以使用pmap
+ awk
。
最有可能的是,我们对以下RSS
示例pmap
输出的最后一行的第三列(82564)的内存感兴趣。
$ pmap -x <pid>
Address Kbytes RSS Dirty Mode Mapping
....
00007f9caf3e7000 4 4 4 r---- ld-2.17.so
00007f9caf3e8000 8 8 8 rw--- ld-2.17.so
00007fffe8931000 132 12 12 rw--- [ stack ]
00007fffe89fe000 8 8 0 r-x-- [ anon ]
ffffffffff600000 4 0 0 r-x-- [ anon ]
---------------- ------ ------ ------
total kB 688584 82564 9592
然后使用Awk提取该值。
$ pmap -x <pid> | awk '/total/ { print $4 "K" }'
该pmap
值以千字节为单位。如果我们希望以兆字节为单位,则可以执行以下操作。
$ pmap -x <pid> | awk '/total/ { print $4 / 1024 "M" }'
采用
例:
ps-of()
{
ps u `pidof "$@"`
}
$ ps-of firefox
USER PID %CPU %MEM VSZ RSS TTY STAT START TIME COMMAND
const 18464 5.9 9.4 1190224 372496 ? Sl 11:28 0:33 /usr/lib/firefox/firefox
$ alias ps-mem="ps xu --sort %mem | sed -e :a -e '1p;\$q;N;6,\$D;ba'"
$ ps-mem
USER PID %CPU %MEM VSZ RSS TTY STAT START TIME COMMAND
const 3656 0.0 0.4 565728 18648 ? Sl Nov21 0:56 /usr/bin/python /usr/lib/ubuntuone-client/ubuntuone-syncdaemon
const 11361 0.3 0.5 1054156 20372 ? Sl Nov25 43:50 /usr/bin/python /usr/bin/ubuntuone-control-panel-qt
const 3402 0.0 0.5 1415848 23328 ? Sl Nov21 1:16 nautilus -n
const 3577 2.3 2.0 1534020 79844 ? Sl Nov21 410:02 konsole
const 18464 6.6 12.7 1317832 501580 ? Sl 11:28 1:34 /usr/lib/firefox/firefox
$ ps h -o pmem -C firefox
12.7