1

我的 Java 应用程序有问题。昨天,当我部署它进行测试运行时,我们注意到我们的机器开始交换,即使这不是一个真正的怪物应用程序,如果你明白我的意思的话。无论如何,我检查了top的结果,发现它消耗了大约 100mb 的内存(顶部的 RES)我试图分析内存并检查是否存在内存泄漏,但我找不到。有一个未关闭的 PreparedStatement,我已修复,但意义不大。我尝试设置最小和最大堆大小(有人说不需要最小堆大小),但没有任何区别。

这就是我现在运行它的方式:

#!/bin/sh

$JAVA_HOME/bin/java -Dcom.sun.management.jmxremote -Dcom.sun.management.jmxremote.port=9025 -Dcom.sun.management.jmxremote.ssl=false -Dcom.sun.management.jmxremote.authenticate=false -XX:MaxPermSize=40m -Xmx32M -cp ./jarName.jar uk.co.app.App app.properties

这是top的结果:

PID USER      PR  NI  VIRT  RES  SHR S %CPU %MEM    TIME+ COMMAND                                                                                                                                                                          
16703 root      20   0  316m 109m 6048 S  0.0 20.8   0:14.80 java   

我不明白的是我配置了最大 PermSize 和最大堆大小,它们加起来是 72mb。够了,应用程序运行良好。为什么它还在吃 109mb 的内存,是什么在吃掉它?这是一个 37mb 的差异,这是一个相当高的比例。(34%)。我不认为这是内存泄漏,因为设置了最大堆大小并且没有内存不足错误或任何东西。

一件有趣的事情可能是我用 VisualVM 做了一个堆转储,然后用 EclipseMAT 检查它,它说类加载器中可能存在泄漏。这就是它所说的:

类加载器/组件“sun.misc.Launcher$AppClassLoader @ 0x87efa40”占用 9,807,664 (64.90%) 个字节。内存在“”.Keywords sun.misc.Launcher$AppClassLoader @ 0x87efa40 加载的“short[][]”的一个实例中累积

我无法充分利用这一点,但可能会有用。

提前感谢您的帮助。

编辑

找到了这个,也许我无能为力... Tomcat内存消耗超过heap + permgen空间

4

2 回答 2

4

Java的内存包括

  • 堆空间。
  • 烫发空间
  • 线程堆栈区域。
  • 共享库,包括 JVM 的库(将被共享)
  • 直接内存大小。
  • 内存映射文件大小(将被共享)

可能还有其他供内部使用。

鉴于 37 MB 的 PC 内存价值约 20 美分,我不会太担心。;)

于 2012-05-25T10:23:03.837 回答
1

您是否尝试使用 JConsole 来分析应用程序http://docs.oracle.com/javase/1.5.0/docs/guide/management/jconsole.html 否则您也可以使用 JProfiler 试用版来分析应用程序 http:// www.ej-technologies.com/products/jprofiler/overview.html?gclid=CKbk1p-Ym7ACFQ176wodgzy4YA

然而,检查高内存使用的第一步应该是检查您是否在应用程序中使用对象集合,如数组、映射、集合、列表等。如果是,则检查它们是否保留对对象的引用(即使未使用)跟他们 ?

于 2012-05-25T10:08:12.147 回答