本篇内容整理自《深入理解JVM》第二章 第四节
本节内容的目的有两个:
- 第一,通过代码验证Java虚拟机规范中描述的长个运行时区域存储的内容;
- 第二,希望读者在工作中遇到实际的内存溢出异常时,能根据异常的信息快速判断是哪个区域的内存溢出,知道什么样的代码可能会导致这些区域内存溢出,以及出现这些异常后该如何处理。
下文的代码都是基于Sun公司的HotSpot虚拟机运行的。
一、Java 堆溢出
Java堆用于存储对象实例,只要不断地创建对象,并且保证GC Roots到对象之间有可达路径来避免垃圾回收机制清除这些对象,那么在对象数量到达最大堆的容量限制后就会产生内存溢出异常。
参数说明:
- -Xms20m:堆的最小值;
- -Xmx20m:堆的最大值。将堆的最小值与最大值设置为一样可避免堆自动扩展;
- -XX:+HeapDumpOnOutOfMemoryError:让虚拟机在出现内存溢出异常时Dump出当前的内存堆转储决照以便事后进行分析。
package testJVM;
import java.util.ArrayList;
import java.util.List;
public class HeapOOM {
static class OOMObject {
}
/**
* VM Args: -Xms20m -Xmx20m -XX:+HeapDumpOnOutOfMemoryError
*/
public static void main(String[] args) {
List<OOMObject> list = new ArrayList<>();
while (true) {
list.add(new OOMObject());
}
}
}
运行代码:
运行结果:
java.lang.OutOfMemoryError: Java heap space
Dumping heap to java_pid904.hprof ...
Heap dump file created [28010942 bytes in 0.086 secs]
Exception in thread "main" java.lang.OutOfMemoryError: Java heap space
at java.util.Arrays.copyOf(Arrays.java:3210)
at java.util.Arrays.copyOf(Arrays.java:3181)
at java.util.ArrayList.grow(ArrayList.java:261)
at java.util.ArrayList.ensureExplicitCapacity(ArrayList.java:235)
at java.util.ArrayList.ensureCapacityInternal(ArrayList.java:227)
at java.util.ArrayList.add(ArrayList.java:458)
at testJVM.HeapOOM.main(HeapOOM.java:18)
刷新一下项目,可以看到Dump出的java_pid904.hprof文件:
接下来我们就可以通过内存映像分析工具(如Eclipse Memory Analyzer)对Dump出来的堆转储快照进行分析。
分析的过程暂时就不讲解了,等待下次在讲解(需要控制每篇博文的长度,避免大家看起来头疼。因为我自己一看到大篇大篇的文字就头疼~_~!)。
二、虚拟机栈和本地方法栈溢出
由于在Hotsput虚拟机中并不区分虚拟机找和本地方法栈,因此,对于HotSpot来说,虽然-Xoss
参数(设置本地方法栈大小)存在,但实际上是无效的,栈容量只由-Xss
参数设定。关于虚拟机栈和本地方法栈,在Java虚拟机规范中描述了两种异常:
- 如果线程请求的栈深度大于虚拟机所允许的最大深度,将抛出StackOverflowError异常;
- 如果虚拟机在扩展栈时无法申请到足够的内存空间,则抛出OutOfMemoryErro:异常。
这里把异常分成两种情况,看似更加严谨,但却存在着一些互相重叠的地方:当栈空间无法继续分配时。到底是内存太小,还是已使用的栈空间太大,其本质上只是对同一件事情的两种描述而已。
在笔者的实验中。将实验范围限制于单线程中的操作,尝试了下面两种方法均无法让虚拟机产生OutOfMemoryError异常,尝试的结果都是获得StackOverfiowError异常,测试代码如下:
- 使用
-Xss
参数减少栈内存容量。结果:抛出StackOvcrflowError异常,异常出现时输出的堆栈深度相应缩小; - 定义了大量的本地变壁,增大此方法帧中本地变最表的长度.结果:抛出StackOverflowError异常时输出的堆栈深度相应缩小。
package testJVM;
public class JavaVMStackSOF {
private int stackLength = 1;
public void stackLeak() {
stackLength++;
stackLeak();
}
/**
* VM Args: -Xss128k
*/
public static void main(String[] args) {
JavaVMStackSOF oom = new JavaVMStackSOF();
try {
oom.stackLeak();
} catch (Throwable e) {
System.out.println("stack length:" + oom.stackLength);
throw e;
}
}
}
Exception in thread "main" stack length:995
java.lang.StackOverflowError
at testJVM.JavaVMStackSOF.stackLeak(JavaVMStackSOF.java:8)
at testJVM.JavaVMStackSOF.stackLeak(JavaVMStackSOF.java:9)
at testJVM.JavaVMStackSOF.stackLeak(JavaVMStackSOF.java:9)
.......
实验结果表明:在单个线程下,无论是由于栈帧太大还是虚拟机栈容最太小,当内存无法分配的时候,虚拟机抛出的都是StackOvcrflowError异常。
如果测试时不限于单线程,通过不断地建立线程的方式倒是可以产生内存溢出异常,如下面的代码所示。但是这样产生的内存溢出异常与找空间是否足够大并不存在任何联系,或者准确地说,在这种情况下,为每个线程的栈分配的内存越大,反而越容易产生内存溢出异常。
其实原因不难理解,操作系统分配给每个进程的内存是有限的,譬如32位的Windows限制为2GB。虚拟机提供了参数来控制Java堆和方法区的这两部分内存的最天值。剩余的内存为2GB(操作系统限制)减去Xmx(最大堆容量),再减去MaxPermSize(最大方法区容量),程序计数器消耗内存很小,可以忽略掉。如果虚拟机进程本身耗费的内存不计算在内,剩下的内存就由虚拟机栈和本地方法栈“瓜分”了。每个线程分配到的栈容量越大,可以建立的线程数量自然就越少,建立线程时就越容易把剩下的内存耗尽。
但是,如果是建立过多线程导致的内存溢出,在不能减少线程数或更换64位虚拟机的情况下,就只能通过减少最大堆和减少栈容量来换取更多的线程。
/**
* VM Args:-Xss2M (这时候不妨设大些)
*/
public class JavaVMStackOOM {
private void dontStop() {
while (true) {
}
}
public void stackLeakByThread() {
while (true) {
Thread thread = new Thread(new Runnable() {
@Override
public void run() {
dontStop();
}
});
thread.start();
}
}
public static void main(String[] args) throws Throwable {
JavaVMStackOOM oom = new JavaVMStackOOM();
oom.stackLeakByThread();
}
}
三、运行时常量池溢出
运行时常量池是方法区的一部分,其实运行时常量池溢出也就是方法区溢出,应该是和第四节放到一起说的,但为了不让大家混淆就分开来说明。
前面提到JDK 1.7开始逐步“去永久代”的事情,在此就以测试代码观察一下这件事对程序的实际影响。
注:下面的一段代码要运行在JDK1.6环境下。
package testJVM;
import java.util.ArrayList;
import java.util.List;
/**
* VM Args:-XX:PermSize=10M -XX:MaxPermSize=10M
*/
public class RuntimeConstantPoolOOM {
public static void main(String[] args) {
// 使用List保持着常量池引用,避免Full GC回收常量池行为
List<String> list = new ArrayList<String>();
// 10MB的PermSize在integer范围内足够产生OOM了
int i = 0;
while (true) {
list.add(String.valueOf(i++).intern());
}
}
}
Exception in thread "main" java.lang.OutOfMemoryError: PermGen space
at java.lang.String.intern(Native Method)
at testJVM.RuntimeConstantPoolOOM.main(RuntimeConstantPoolOOM.java:17)
String.intern()是一个Native方法,它的作用是:如果字符串常量池中已经包含一个等于此String对象的字符串,则返回代表池中这个字符串的String对象;否则,将此String对象包含的字符串添加到常量池中,并且返回此String对象的引用。在JDK 1.6及之前的版本中,由于常量池分配在水久代内,我们可以通过
-XX:PermSize
和-XX:MaxPermSize
限制方法区大小,从而间接限制其中常量池的容量。
从运行结果中可以看到,运行时常量池溢出,在OutOfMcmoryError后面跟随的提示信息是“PermGen space”,说明运行时常量池属于方法区(HotSpot虚拟机中的永久代)的一部分。而使用JDK 1.7运行这段程序就不会得到相同的结果,while循环将一直进行下去。
四、方法区溢出
方法区用于存放Class的相关信息,如类名、访问修饰符、常量池、字段描述、方法描述等。对于这些区域的测试,基本的思路是运行时产生大量的类去填满方法区,直到溢出。代码中借助CGLib直接操作字节码运行时生成了大量的动态类。
package testJVM;
import java.lang.reflect.Method;
import net.sf.cglib.proxy.Enhancer;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;
public class JavaMethodAreaOOM {
/**
* VM Args: -XX:PermSize=10M -XX:MaxPermSize=10M
*/
public static void main(String[] args) {
while (true) {
Enhancer enhancer = new Enhancer();
enhancer.setSuperclass(OOMObject.class);
enhancer.setUseCache(false);
enhancer.setCallback(new MethodInterceptor() {
public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {
return proxy.invokeSuper(obj, args);
}
});
enhancer.create();
}
}
static class OOMObject {
}
}
Caused by: java.lang.OutOfMemoryError: PermGen space
at java.lang.ClassLoader.defineClass1(Native Method)
at java.lang.ClassLoader.defineClass(ClassLoader.java:791)
... 8 more
五、本机直接内存溢出
DirectMemory容量可通过-XX : MaxDirectMemorySize
指定,如果不指定,则默认与Java堆最大值(-Xmx指定)一样,下面代码越过了DirectByteBuffer类,直接通过反射获取Unsafe实例进行内存分配(Unsafe类的getUnsafe()方法限制了只有引导类加载器才会返回实例,也就是设计者希望只有rt.jar中的类才能使用Unsafe的功能)。因为,虽然使用DirectByteBuffer分配内存也会抛出内存溢出异常,但它抛出异常时并没有真正向操作系统申请分配内存,而是通过计算得知内存无法分配,于是手动抛出异常,真正申请分配内存的方法是unsafe.allocateMemory()。
package testJVM;
import java.lang.reflect.Field;
import sun.misc.Unsafe;
public class DirectMemoryOOM {
private static final int _1MB = 1024 * 1024;
/**
* VM Args:-Xmx20M -XX:MaxDirectMemorySize=10M
*/
public static void main(String[] args) throws Exception {
Field unsafeField = Unsafe.class.getDeclaredFields()[0];
unsafeField.setAccessible(true);
Unsafe unsafe = (Unsafe) unsafeField.get(null);
while (true) {
unsafe.allocateMemory(_1MB);
}
}
}
Exception in thread "main" java.lang.OutOfMemoryError
at sun.misc.Unsafe.allocateMemory(Native Method)
at testJVM.DirectMemoryOOM.main(DirectMemoryOOM.java:19)
由DirectMemory导致的内存溢出,一个明显的特征是在Heap Dump文件中不会看见明显的异常。如果读者发现OOM之后Dump文件很小,而程序中又直接或间接使用了NIO,那就可以考虑检查一下是不是这方面的原因。