为什么是锯齿形图?

当我使用NetBeans运行下面提到的代码时,分配的堆大小图类似于锯齿形状。 我正在附加来自JVisualVM的屏幕截图,它显示了具有锯齿形状的堆分配图。 该程序是一个简单的无限循环打印“Hello,World!” 进入控制台。

public class HelloWorld { public static void main(String a[]){ while(true) { System.out.println("Hello, World!"); } } } 

在此处输入图像描述 任何人都可以解释使用堆图形状背后的原因吗?

PS :即使我在不​​使用NetBeans的情况下运行它也会发生这种情况,因此它很可能与NetBeans无关…

堆使用中的锯齿模式可以通过在调用System.out.println调用期间创建几个局部变量的事实来解释。 最值得注意的是在Oracle / Sun JRE中,在年轻代中创建了几个HeapCharBuffer实例,如使用VisualVM的内存分析器获得的以下快照中所述:

Visual VM  - 内存快照

有趣的是在堆上存在的活动对象的数量。 锯齿形图案是由伊甸园空间填满时发生的年轻垃圾收集周期产生的; 由于在程序中没有执行繁重的计算活动,JVM能够执行循环的多次迭代,从而导致伊甸园空间(大小为4MB)填满。 随后的年轻人收集周期然后清除大部分垃圾; 它几乎总是整个伊甸园空间,除非对象仍在使用中,如从VisualVM获得的以下gc轨迹所示:

Visual VM GC探针

因此,锯齿形图案的行为可以通过一系列快速连续的物体分配来解释,这些物体分配填满了伊甸园空间,触发了年轻的垃圾收集循环; 由于底层JVM进程没有被另一个进程抢占,因此这个进程不间断地重复循环,并且JVM中负责对象分配的主线程也不会被另一个线程抢占。

以常规速率分配对象的任何进程都将导致堆内存消耗的稳定增加,然后在垃圾收集器收集不再使用的对象时立即下降,从而产生锯齿形状。

如果你想知道为什么你的java进程在写入System.out保持分配内存,请记住其他线程(例如将当前内存统计信息提供给JVisualVM的线程)可能是分配内存的线程。

它可能来自很多地方,并且可能依赖于实现。 至少以下是可能的(但都只是猜测)

  • 在System.out.println下面的流堆栈中的某处有一个字节数组分配(假设输出流的一个基本方法是write(bytes [] b,int off,int len))

  • 它是您正在使用的监控软件使用的开销(我没有使用它)

  • 它是netbeans VM中的开销,它最终显示输出

实际上jVisualVM导致额外的对象分配。 jVisualVM和jconsole正在使用Java Management Extensions。 附加到正在运行的应用程序并请求JVM指标,从而导致正在创建其他对象。 您可以通过添加到您的程序调用来validation这一点

 Runtime.getRuntime().freeMemory() 

报告JVM堆中的可用内存。 它将通过运行代码显示[几乎]没有内存更改,但只要将jVisualVM连接到程序,您就会看到内存使用量增加。