除程序计数器外,虚拟机内存的其它几个运行时区域都有可能发生OutOfMemoryError
1、Java堆溢出
Java堆用于储存对象实例,我们只用不断的创建对象,并且保证GC Roots到对象之间没有可达路径来避免垃圾回收清除这些对象,如此当对象数量到达最大堆的容量限制后产生内存溢出
/**
* 模拟堆内存溢出
* VM Args -Xms20m -Xmx20m -XX:+HeapDumpOnOutOfMemoryError(可以让虚拟机在出现内存溢出异常时Dump出当前的内存堆转储快照以便事后分析)
*/
public static void outOfMemary(){
List<OOMObject> list = new ArrayList<>();
while (true){
list.add(new OOMObject());
}
}
2、虚拟机栈和本地方法栈溢出
每次调用方法都会创建一个栈帧,此处使用递归不断调用方法创建栈帧,
(1)如果线程请求的栈深度大于虚拟所允许的最大深度,则会抛出StackOverflowError
(2)如果虚拟机在扩展栈时无法申请到足够的内存空间,则抛出OutOfMemoryError
private static int satckLength = 1;
/**
* 模拟栈内存溢出
* VM Args -Xss128k
*/
public static void stackOverFlow(){
satckLength++;
stackOverFlow();
}
ps:操作系统分配给每个进程的内存是有限的,如果建立过多线程导致的内存溢出,在不减少线程或更换64位操作虚拟机的情况下,就只能通过减少最大堆和减少栈容量来换取更多的线程。
3、运行时常量池溢出
运行时常量池位于方法区中,保存的是字面量和引用,我们可以使用String.intern()这个Native方法,该方法的作用是:当池中已经包含一个等于此String对象的字符串,则返回代表池中这个字符串的对象;否则将次String对象包含的字符串添加到池中,并且返回此String的引用
/**
* 模拟运行时常量池溢出
* VM Args -XX:PermSize=10m -XX:MaxPermSize=10m
*/
public static void runtimeConstantPoolOOM(){
List<String> list = new ArrayList<>();
int i = 0;
while (true){
list.add(String.valueOf(i++).intern());
}
}