使用 javap 深入理解 Java 整型常量和整型变量的区别

我下图代码第五行和第九行分别定义了一个整型变量和一个整型常量:
static final int number1 = 512;
static int number3 = 545;
Java 程序员都知道两者的区别。

下面我们就用 javap 将.class 文件反编译出来然后深入研究 Java 里整型变量和整型常量的区别。
使用命令行 javap -c constant.ConstantFolding 查看.class 文件反编译出来的字节码:

结果:

这些字节码指令的说明,在 wikipedia 里有说明:
wiki: https://en.wikipedia.org/wiki/Java_bytecode_instruction_listings
咱们 Java 程序员不需要把它们都背下来,只需要把这个网页收藏起来,要用的时候当成字典来用就行:

sipush 545: 将整数 545 放置到栈上
putstatic #16:
将栈上的值 545 赋给当前类的静态字段里。

那么 putstatic #16 里的 #16 代表什么含义?
我们再用 javap -v 参数反编译,就能看到这个类的常量池(Constant pool). 大家看下图蓝色高亮的一行:
constant/ConstantFolding.number3:I
说明 #16 代表类 constant.ConstantFolding 的成员 number3,类型为 I。

至此,这两行字节码指令联合起来,实际对应了我们写的 Java 代码:
static int number3 = 545;
我们继续分析 javap 反编译出来的字节码。

aload_0: 将序号为 0 的本地变量的引入加载到栈上
invokespecial: 调用对象实例上的成员方法,如果有返回值,方法的返回值存储到栈上。具体调用的方法由 #标识,可在常量池中查询到对应的方法名。
ldc: 将常量池上代号为 #<数字>的常量的值从常量池加载到栈上。
我们从下图的常量池列表能发现,序号为 #29 的常量 318976 正是整型常量 number1(512)和整型常量(623)的积。由此可以看出, number1 * number2 这个表达式,因为参与运算的两个操作数通过 STATIC 和 FINAL 修饰成为了整型常量,因此其积在编译期就能得到,所以编译器在编译时就计算出来,存储在变量池里,序号为 #29。

那么整型变量做乘法运算,对应的字节码又是什么样的呢?
从下图序号为 3 的 code 开始:
getstatic #16: 将类的静态成员 #16 加载到栈上。#16 对应的成员为 number3,值为 545。
getstatic #18: 将类的静态成员 #18 加载到栈上。#18 对应的成员为 number4,值为 619。
imul: 执行栈上两个整数的乘法运算。
istore_2: 将结果保存到局部变量 2 里。
此时,我们 Java 代码里的 int product2 = number3 * number4 就执行完了。
大家看到的剩下的蓝色字节码,都对应了下面这行打印语句。
System.out.println("Value: " + product1 + " , " + product2);
从这些字节码也能看出,Java 里我们直接用加号进行字符串拼接操作,Java 编译器在编译时,自动使用了 StringBuilder 进行优化。
既然整型变量的乘积需要打印出来,因此字节码的 iload_2 将之前用 istore_2 保存在局部变量 2 中的计算结果又加载到栈上,这样乘积结果最后就能输出了。

希望通过这个简单的例子,大家能学会用 javap 去深入理解一些 Java 和 JVM 的细节。
要获取更多 Jerry 的原创技术文章,请关注公众号"汪子熙"。
版权声明: 本文为 InfoQ 作者【Jerry Wang】的原创文章。
原文链接:【http://xie.infoq.cn/article/b24544d7308c4b7a41153d886】。文章转载请联系作者。
评论