3

这个问题专门针对在 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来调查问题。

4

3 回答 3

3

我认为您的 Eclipse 环境的高内存消耗的原因是使用了 SWT。SWT 是一个位于 JVM 堆之外的本地图形库,更糟糕的是,Linux 上的实现并没有真正优化。

我不认为真的有机会减少关于堆外内存的 Eclipse 环境的内存消耗。

于 2011-10-18T10:31:53.473 回答
1

Eclipse 是内存和 cpu 猪。除了 Java 类库之外,所有低端 GUI 内容都由本机系统调用处理,因此您将拥有一个实质性的“本机”JNI 库来执行附加到您的进程的低级 X 术语调用。

Eclipse 提供了数以百万计的有用功能和大量帮助程序来加快您的日常编程任务——但精益求精并意味着它不是。内存或资源的任何减少都可能导致明显的减速。这实际上取决于您对时间与计算机内存的重视程度。

如果你想要精益求精,那么 gvim 和 make 是无与伦比的。如果您想要代码完成、自动构建等,您必须期望为此付出额外的资源。

于 2011-10-18T10:32:37.673 回答
0

如果我运行以下程序

public static void main(String... args) throws InterruptedException {
    for (int i = 0; i < 60; i++) {
        System.out.println("waiting " + i);
        Thread.sleep(1000);
    }
}

ps auwx印花

USER       PID %CPU %MEM    VSZ   RSS TTY      STAT START   TIME COMMAND
500      13165  0.0  0.0 596680 13572 pts/2    Sl+  13:54   0:00 java -Xms64m -Xmx200m -XX:NewSize=8m -XX:PermSize=80m -XX:MaxPermSize=150m -cp . Main

使用的内存量为 13.5 MB。大约有 200 MB 的共享库计入 VSZ 大小。其余的可以在最大堆、最大 perm gen 中计算,以及线程堆栈的开销等。

问题似乎不在于 JVM,而在于在其中运行的应用程序。使用额外的共享库、直接内存和内存映射文件可以增加使用的内存量。

假设您可以以 100 美元左右的价格购买 16 GB,您知道这实际上是个问题吗?

于 2011-10-18T13:02:45.677 回答