今天看到一个很有意思的提问:(a+b)*10
,10
是存放在哪里的?是常量池么?如果是常量池,在进行运算的时候,是通过指针来找到的吧?
某回答:10
是在常量池,常量池在jdk1.8
以后已经移到元空间了。
要验证这个答案是否正确其实很简单,写一个测试方法通过javap
查看编译后的字节码指令就能得出结论。
测试方法源码如下:
private int m(int a, int b) {
int r = a * b * 10;
return r;
}
使用javap
查看编译后的字节码如下:
private int m(int, int);
descriptor: (II)I
flags: ACC_PRIVATE
Code:
stack=2, locals=4, args_size=3
0: iload_1
1: iload_2
2: imul
3: bipush 10
5: imul
6: istore_3
7: iload_3
8: ireturn
笔者翻阅了一下JVM
字节码指令表,相关指令如下图所示。
字节码指令由操作码和零个或多个操作数组成,编译后会被写入方法的Code
属性中,操作数要么是立即数,要么是指向class
文件结构常量池中常量的索引,要么是跳转目标指令的偏移量。
设立即数10
为x
,可得出以下结论:
x
取值在区间[-1,5]
时,x
直接被编译进方法的code
属性中,作为iconst
指令的操作数;x
取值在区间[-128,127]
时,x
也直接被编译进方法的code
属性中,作为bipush
指令的操作数,取值范围[-128,127]
是因为bipush
指令的操作数只能是一个字节;x
取值在区间[-32768,32767]
时,x
也直接被编译进方法的code
属性中,作为sipush
指令的操作数,取值范围[-32768,32767]
是因为sipush
指令的操作数有两个字节;x
编译进class
的常量池,在进行运算时通过ldc
指令从常量池(元空间)push
进操作数栈中。