先了解一下Java堆:
关于Java内存区域的分配,可以查看Java运行时数据区域一篇文章。
Java堆是虚拟机内存管理中最大的一块区域,该区域是线程共享的,某Java进程中所有的线程都可以访问该区域中的数据。Java堆中大部分区域是存放的实例对象,就是我们常说的Java面向对象编程中的对象。比如我创建了一个类,要通过这个类的构造函数new一个对象,new出来的这个对象就是存放在Java堆上。根据对象产生时间和引用时间的长短,可以细分为新生代和老年代。
通过-Xms和-Xmx参数控制虚拟机启动时Java堆的大小。如果实例对象没有在Java堆上分配,并且堆无法扩展,将会抛出OutOfMemoryError异常。
接下来模拟一下Java堆的溢出。以下是测试代码:
package com.woasis.jvm.classloading; import java.util.ArrayList;
import java.util.List; /**
* 模拟堆溢出
* -Xms20M 最小初始内存
* -Xmx20M 最大内存值
* -XX:+HeapDumpOnOutOfMemoryError
*/
public class DeapOutOfMemory { /**
* 内部类,用于创建对象,存于堆内存中
*/
static class TestOOM{ } public static void main(String[] args) { //用于存放模拟的对象,防止GC回收,用List做对象引用
List<TestOOM> list = new ArrayList<>(); for (;;){
list.add(new TestOOM());
}
}
}
便于模拟溢出,和抓取内存溢出时的内存快照,如下图设置虚拟机参数:
-Xms20M Java堆内存初始化大小为20M
-Xmx20M Java堆内存最大值为20M
-XX:+HeapDumpOnOutOfMemoryError 当发生OutOfMemoryError异常时抓取Java堆快照。其中 + 是打开的意思。
设置好虚拟机内存参数后运行Main方法。如下图出现OutOfMemoryError异常,发生的地方是Java Heap Space。
对抓取的快照进行打开,如下图,可以看到内部类TestOOM实例对象数量是810326。