由于在HotSpot虚拟机中并不区分虚拟机栈和本地方法栈,因此对于HotSpot来说,-Xoss参数(设置本地方法栈大小)虽然存在,但实际上是无效的,栈容量只由-Xss参数设定。关于虚拟机栈和本地方法栈,在Java规范中描述了两种异常:
- 如果线程请求的栈深度大于虚拟机所允许的最大深度,将抛出StackOverflowError异常。
- 如果虚拟机在扩展栈时无法申请到足够的内存空间,则抛出OutOfMemoryError异常。
在下面的代码中,如果将范围限制于单线程中的操作,尝试下面两种方法均无法让虚拟机产生OutOfMemoryError异常,尝试的结果都是获得StackOverflowError异常,方法如下:
使用-Xss参数减少栈内存容量。结果抛出StackOverflowError异常,异常出现时输出的栈深度相应的缩小。
StackOverflowError异常代码如下:
package oom; /** * 虚拟机栈和本地方法栈溢出 * @author Madison * @date 2014-7-11 * VM Args:-Xss128k */ public class JavaVMStackSOF { private int stackLength = 1; public void stackLeak() { stackLength++; stackLeak(); } public static void main(String[] args) throws Throwable { JavaVMStackSOF oom = new JavaVMStackSOF(); try { oom.stackLeak(); } catch(Throwable e) { System.out.println("stack length:" + oom.stackLength); throw e; } } }
运行结果:
stack length:2402
Exception in thread "main" java.lang.StackOverflowError
at oom.JavaVMStackSOF.stackLeak(JavaVMStackSOF.java:15)
at oom.JavaVMStackSOF.stackLeak(JavaVMStackSOF.java:15)
......后续异常栈信息省略
通过代码运行结果表明:在单个线程下,无论是由于栈帧太小,还是虚拟机栈容量太小,内存无法分配的时候,虚拟机抛出的都是StackOverflowError异常。
如果测试时不限于单线程,通过不断地建立线程的方式倒是可以产生内存溢出异常,但是,这样产生的内存溢出与栈空间是否足够大并不存在任何联系,或者准确地说,在这种情况下,给每个线程的栈分配的内存越大,反而容易产生内存溢出异常。
操作系统分配给每个进程的内存是有限制的。譬如32位Windows限制为2GB。虚拟机提供了参数来控制Java堆和方法区的这两部分内存的最大值。剩余的内存为2GB(操作系统限制)减去Xmx(最大堆容量),再减去MaxPermSize(最大方法区容量),程序计数器消耗的内存很小,可以忽略掉。如果虚拟机进程本身耗费的内存不计算在内,剩下的内存就由虚拟机栈和本地方法栈“瓜分”了。每个线程分配的栈容量越大,可以建立的线程数量自然就越少,建立线程时就越容易把剩下的内存耗尽。
这一点需要在开发多线程应用的时候特别注意,出现StackOverflowError异常时有错误堆栈可以阅读,相对来说,比较容易找到问题的所在。而且,如果是使用虚拟机默认参数,栈深度在大多数情况下达到1000~2000完全没有问题,对于正常的方法调用,这个深度应该完全够用了。但是,如果建立过多线程导致的内存溢出,在不减少线程数或者更换64位虚拟机的情况下,就只能通过减少最大堆和减少栈容量来换取更多的线程。如果没有这方面的经验,这种通过“减少内存”的手段来解决内存溢出的方式会比较难以想到。
创建线程导致内存溢出异常代码如下,运行该代码可能会导致操作系统假死。
package oom; /** * 创建线程导致内存溢出 * @author Madison * @date 2014-7-11 * 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) { JavaVMStackOOM oom = new JavaVMStackOOM(); oom.stackLeakByThread(); } }
运行结果:
Exception in thread "main" java.lang.OutOfMemoryError:unable to create new native thread
欲知后事如何,且听下回分解
JVM【第六回】:【OutOfMemoryError异常之虚拟机栈和本地方法栈溢出】