每个栈帧中存储着:
局部变量表存放了编译期可知的各种Java虚拟机基本数据类型(boolean、byte、char、short、int、float、long、double)、对象引用(reference类型,它并不等同于对象本身,可能是一个指向对象起始地址的引用指针,也可能是指向一个代表对象的句柄或者其他与此对象相关的位置)和returnAddress 类型(指向了一条字节码指令的地址)。
=====================================
这些数据类型在局部变量表中的存储空间以局部变量槽(Slot)来表示,其中64位长度的long和double类型的数据会占用两个变量槽,其余的数据类型只占用一个。局部变量表所需的内存空间在编译期间完成分配,当进入一个方法时,这个方法需要在栈帧中分配多大的局部变量空间是完全确定的,在方法运行期间不会改变局部变量表的大小。请读者注意,这里说的“大小”是指变量槽的数量,虚拟机真正使用多大的内存空间(譬如按照1个变量槽占用32个比特、64个比特,或者更多)来实现一个变量槽,这是完全由具体的虚拟机实现自行决定的事情。
=====================================
在《Java虚拟机规范》中,对这个内存区域规定了两类异常状况:如果线程请求的栈深度大于虚拟机所允许的深度,将抛出StackOverflowError异常;如果Java虚拟机栈容量可以动态扩展[2],当栈扩展时无法申请到足够的内存会抛出OutOfMemoryError异常。
--------摘自《深入理解java虚拟机》
对于局部变量表所需的容量大小是在编译期确定下来的这句话可以通过看字节码文件
源代码:
public class Example {
public static void main(String[] args) {
int a = 3;
a++;
testStatic();
System.out.println(a);
}
public static void testStatic(){
Date date = new Date();
int count = 10;
System.out.println(count);
}
}
字节码:
可以看到 locals=2 说明了局部变量表的大小为2 ,而这两个变量为data和count,所以说局部变量表所需的容量大小是在编译期确定下来的。(此时代码只是进行了编译还未运行)
通过使用jclasslib来看字节码,进行一些相关解释。
1.字节码行号
字节码中左边的数字表示的是有多少行字节码0~15也就是有16行。
2.方法异常信息表
此为异常信息表,当前方法没有异常所以没有异常表。
3、Misc(杂项)
4、行号表
Java代码的行号和字节码指令行号的对应关系
5、生效行数和剩余有效行数(针对于字节码文件的行数)
图中标记的地方表示的是该局部变量的作用域,初始PC(Start PC)为2表示该局部变量在字节码的第2行开始生效,字节码的第2行对应着java代码的第8行(由上一张图可知),而int a的定义是在第7行,可以得知局部变量是从声明的下一行生效的。
长度(Length)表示剩余有效行数,main方法字节码指令总共有16行,从2行开始生效,那么剩下就是16-2 =14。
描述符(Descriptor)第一行 [Ljava/lang/String 表示args的引用类型(String[]),第二行 I 表示的是a的引用类型(int)
public class Example {
public int sum = 0;
public static void main(String[] args) {
new Example().test();
}
public void test() {
this.sum++;
double a = 3;
long b = 4;
}
}
注意:
栈帧中的局部变量表中的槽位是可以重用的,如果一个局部变量过了其作用域,那么在其作用域之后申明新的局部变量变就很有可能会复用过期局部变量的槽位,从而达到节省资源的目的。
public void test() {
int a = 0;
{
int b = 0;
b = a + 1;
}
//变量c使用之前已经销毁的变量b占据的slot的位置
int c = a + 1;
}
可以看到局部变量c重用了局部变量b的slot位置
变量的分类:
变量的赋值:
原文:https://www.cnblogs.com/niulongwei/p/14864516.html