java - JVM以编程方式获取堆中最大的对象

标签 java jvm profiling heap-memory profiler

我如何以编程方式(从 java 应用程序/代理中)获得堆中最大对象的“实时”摘要(包括它们的实例数和大小)?

类似于分析器所做的。

例如,这是 JProfiler 的屏幕截图:

Live objects

通常我在真正需要堆转储的情况下使用堆转储,但现在我想弄清楚探查器究竟如何从运行中的 JVM 中检索此类信息,而无需实际进行堆转储。

是否可以使用Java API获取此类信息?如果不可能, native 代码中的替代方案是什么?代码示例最能满足我的需求,因为 Java 世界的这个特定部分对我来说真的很陌生。

我有点相信,如果我有兴趣找到关于某些真正特定类的信息,我可以使用仪器或其他东西,但据我所知,它使用采样,所以应该有另一种方法。

我目前正在使用在 Linux 上运行 Java 8 的 HotSpot VM,但是我会找到越“通用”的解决方案 - 越好。

最佳答案

堆遍历没有标准的 Java API。但是,有一个特定于 HotSpot 的诊断命令可以通过 JMX 调用:

String histogram = (String) ManagementFactory.getPlatformMBeanServer().invoke(
        new ObjectName("com.sun.management:type=DiagnosticCommand"),
        "gcClassHistogram",
        new Object[]{null},
        new String[]{"[Ljava.lang.String;"});

这将收集类直方图并将结果作为单个格式化的 String 返回:

 num     #instances         #bytes  class name
----------------------------------------------
   1:          3269         265080  [C
   2:          1052         119160  java.lang.Class
   3:           156          92456  [B
   4:          3247          77928  java.lang.String
   5:          1150          54104  [Ljava.lang.Object;
   6:           579          50952  java.lang.reflect.Method
   7:           292          21024  java.lang.reflect.Field
   8:           395          12640  java.util.HashMap$Node
                       ...

内容等同于jmap -histo命令的输出。

堆遍历的唯一标准 API 是 native JVM TI IterateThroughHeap 函数,但它不是那么容易使用,而且它的运行速度比上述诊断命令慢得多。

关于java - JVM以编程方式获取堆中最大的对象,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/67148994/

相关文章:

java - 合并c :out tag with bean:message

java - Spring OS 依赖映射

java - Linux操作系统中的JVM堆参数

linux - 用于识别最易受内存性能影响的应用数据/代码的工具

java - 如何在 Mac 上安装旧版本的 JDK?

java - 从文件属性中检索标签列表

profiling - perf.data 文件没有样本

java - 使用 JVMTI 获取 GC 释放的内存量

java - 增加 maxpermsize 和最大堆大小的副作用

java - Java中如何查找一个对象的所有引用者?