1.局部变量表: 存放了编译期可知的各种基本类型、对象引用和returnAddress类型(指向了一条字节码指令地址)。其中64位长度long 和 double占两个局部变量空间,其他只占一个。值得注意的是:局部变量表所需的内存空间在编译期间完成分配。在方法运行的阶段是不会改变局部变量表的大小的。
2.操作数栈: 虚拟机把操作数栈作为它的工作区,程序中的所有计算过程都是在借助于操作数栈来完成的,大多数指令都要从这里弹出数据,执行运算,然后把结果压回操作数栈。
3.动态连接: 每个栈帧都包含一个指向运行时常量池中该栈帧所属方法的引用(指向运行时常量池:在方法执行的过程中有可能需要用到类中的常量),持有这个引用是为了支持方法调用过程中的动态连接。
4.方法出口: 当一个方法执行完毕之后,要返回之前调用它的地方,因此在栈帧中必须保存一个方法返回地址。
规定的异常情况有两种:1.线程请求的栈的深度大于虚拟机所允许的深度,将抛出StackOverflowError异常;2.如果虚拟机可以动态扩展,如果扩展时无法申请到足够的内存,就抛出OutOfMemoryError异常。
5.附加信息: 虚拟机规范允许具体的虚拟机实现增加一些规范里没有描述的信息到栈帧中,例如与高度相关的信息,这部分信息完全取决于具体的虚拟机实现。在实际开发中,一般会把动态连接,方法返回地址与其它附加信息全部归为一类,称为栈帧信息。
设置JVM参数”-Xss228k”(栈大小为228k)。没有终止条件的递归,执行结果如下图所示,JVM抛出StackOverflowError表示线程请求的栈深度大于JVM所允许的深度。
/**
* -Xss228k,虚拟机栈大小为228k
* Created by yulinfeng on 7/11/17.
*/
public class Test {
private static int count = 0;
public static void main(String[] args) {
Test test = new Test();
test.test();
}
/**
* 递归调用
*/
private void test() {
try {
count++;
test();
} catch (Throwable e) { //Exception已经捕获不了JVM抛出的StackOverflowError
System.out.println("递归调用次数" + count);
e.printStackTrace();
}
}
}
import java.util.ArrayList;
import java.util.List;
/**
* -Xms20M -Xmx20M 堆初始大小20M 堆最大大小20M
* Created by yulinfeng on 7/11/17.
*/
public class Test {
public static void main(String[] args) {
List list = new ArrayList();
int count = 0;
try {
while (true) {
count++;
list.add(new Test()); //不断创建线程
}
} catch (Throwable e) {
System.out.println("创建实例个数:" + count);
e.printStackTrace();
}
}
}
import java.util.ArrayList;
import java.util.List;
/**
* -XX:MaxPermSize=20M 方法区最大大小20M
* Created by yulinfeng on 7/11/17.
*/
public class Test {
public static void main(String[] args) {
List list = new ArrayList();
int i = 0;
while (true) {
list.add(String.valueOf(i++).intern()); //不断创建线程
}
}
}
实际上对于以上代码,在JDK6、JDK7、JDK8运行结果均不一样。原因就在于字符串常量池在JDK6的时候还是存放在方法区(永久代)所以它会抛出OutOfMemoryError:Permanent Space;而JDK7后则将字符串常量池移到了Java堆中,上面的代码不会抛出OOM,若将堆内存改为20M则会抛出OutOfMemoryError:Java heap space;至于JDK8则是纯粹取消了方法区这个概念,取而代之的是”元空间(Metaspace)“,所以在JDK8中虚拟机参数”-XX:MaxPermSize”也就没有了任何意义,取代它的是”-XX:MetaspaceSize“和”-XX:MaxMetaspaceSize”等。
当方法区无法满足内存分配需求时,抛出OutOfMemoryError
import sun.nio.ch.DirectBuffer;
import java.nio.ByteBuffer;
public class Main {
public static void main(String[] args) throws InterruptedException {
System.out.println("Hello World!");
ByteBuffer bb = ByteBuffer.allocateDirect(1024 * 1024 * 128);
Thread.sleep(10000);
((DirectBuffer)bb).cleaner().clean();
Thread.sleep(10000);
}
}