这个问题专门针对在 Linux x86-64 上运行的 Sun Java JVM。我试图弄清楚为什么即使我设置了 Heap 和 Non-Heap 限制,Sun JVM 也会占用这么多系统的物理内存。
我正在运行的程序是具有多个插件/功能的 Eclipse 3.7。最常用的功能是 PDT、EGit 和 Mylyn。我正在使用以下命令行开关启动 Eclipse:
-nosplash -vmargs -Xincgc -Xms64m -Xmx200m -XX:NewSize=8m -XX:PermSize=80m -XX:MaxPermSize=150m -XX:MaxPermHeapExpansion=10m -XX:+UseConcMarkSweepGC -XX:CMSInitiatingOccupancyFraction=70 -XX:+UseCMSInitiatingOccupancyOnly -XX:+UseParNewGC -XX:+CMSIncrementalMode -XX:+CMSIncrementalPacing -XX:CMSIncrementalDutyCycleMin=0 -XX:CMSIncrementalDutyCycle=5 -XX:GCTimeRatio=49 -XX:MaxGCPauseMillis=50 -XX:GCPauseIntervalMillis=1000 -XX:+UseCMSCompactAtFullCollection -XX:+CMSClassUnloadingEnabled -XX:+DoEscapeAnalysis -XX:+UseCompressedOops -XX:+AggressiveOpts -Dorg.eclipse.swt.internal.gtk.disablePrinting
值得注意的是开关:
-Xms64m -Xmx200m -XX:NewSize=8m -XX:PermSize=80m -XX:MaxPermSize=150m
这些开关应将 JVM 堆限制为最大 200 MB,将非堆限制为 150 MB(JConsole 标记的“CMS 永久生成”和“代码缓存”)。从逻辑上讲,JVM 应该占用 350 MB 加上 JVM 所需的内部开销。
实际上,JVM为我当前的 Eclipse 进程占用 544.6 MB,由 ps_mem.py ( http://www.pixelbeat.org/scripts/ps_mem.py ) 计算,它计算 Linux 2.6+ 内核保留的真实物理内存页. 那是 35% 或大约 200MB 的内部 Sun JVM 开销!
关于如何减少这种开销的任何提示?
以下是一些附加信息:
$ ps auxw
USER PID %CPU %MEM VSZ RSS TTY STAT START TIME COMMAND
me 23440 2.4 14.4 1394144 558440 ? Sl Oct12 210:41 /usr/bin/java ...
根据 JConsole,该进程使用了 160 MB 的堆和 151 MB 的非堆。
我并不是说我不能使用额外的 200MB 来运行 Eclipse,但如果有办法减少这种浪费,我宁愿将 200MB 用于内核块设备缓冲区或文件缓存。此外,我对其他 Java 程序也有类似的经验——也许我可以通过类似的调整来减少所有这些程序的开销。
更新:发布问题后,我发现之前的帖子 SO:
为什么即使堆等大小稳定,Sun JVM 仍会继续消耗更多的 RSS 内存?
看来我应该用它pmap
来调查问题。