代码实例演示:
经过反编译后生成的字节码指令
可以看出
i**:操作数类型为int
byte类型的数据在进操作数栈和存储时都是int类型
原因:
在局部变量表里,32位以内的类型只占用一个slot,64的类型占用两个slot。
对于64位的类型数据,java虚拟机会以高位对其的方式分配2个连续的变量槽空间
在操作数栈中大多也是32位为一个单元。
补充:
java虚拟机规范并没有指出变量槽的具体占用空间,只是每个变量槽都应该能存放一个boolean,byte,char,short,int,float,reference和returnAddress类型的数据.这8种数据类型都是以32位或者更小的内存来存储的.但这种描述和每个变量槽用32位来存储是有差别的,它允许变量槽随着环境的变化而变化.
Java每个方法在JVM中对应一个虚拟机栈中的栈帧
方法的形参,局部变量,对象引用和返回值类型存在局部变量表中。
局部变量表相对于一个数组。
局部变量表中最基本的存储单元就是Slot。
在局部变量表里,32位以内的类型只占用一个slot,64的类型占用两个slot。
同理,在操作数栈中,一个数组索引对应的单元为32位。
为什么JVM局部变量表的一个slot至少要能容纳一个int类型的变量?
Java语言被创造出来的时候显然受到了SUN公司自己的SPARC精简指令集的设计影响,同时也为了满足在网络上传输代码的需求,Java虚拟机的字节码指令被限定为一个字节,因此操作码总数不能超过256个,这迫使字节码指令只能选择支持一部分数据类型。这个设计上的选择带来了一些side effect,其中就包括局部变量表和操作数栈的slot大小。