当前位置:主页 > 查看内容

(a+b)*10,10是存在哪里的?是常量池么?

发布时间:2021-04-28 00:00| 位朋友查看

简介:今天看到一个很有意思的提问:(a+b)*10,10是存放在哪里的?是常量池么?如果是常量池,在进行运算的时候,是通过指针来找到的吧? 某回答:10是在常量池,常量池在jdk1.8以后已经移到元空间了。 要验证这个答案是否正确其实很简单,写一个测试方法通过javap查……

今天看到一个很有意思的提问:(a+b)*10,10是存放在哪里的?是常量池么?如果是常量池,在进行运算的时候,是通过指针来找到的吧?

某回答:10是在常量池,常量池在jdk1.8以后已经移到元空间了。

要验证这个答案是否正确其实很简单,写一个测试方法通过javap查看编译后的字节码指令就能得出结论。

测试方法源码如下:

  1. private int m(int a, int b) { 
  2.     int r = a * b * 10; 
  3.     return r; 

使用javap查看编译后的字节码如下:

  1. private int m(intint); 
  2.   descriptor: (II)I 
  3.   flags: ACC_PRIVATE 
  4.   Code: 
  5.     stack=2, locals=4, args_size=3 
  6.          0: iload_1 
  7.          1: iload_2 
  8.          2: imul 
  9.          3: bipush        10 
  10.          5: imul 
  11.          6: istore_3 
  12.          7: iload_3 
  13.          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进操作数栈中。

本文转载自微信公众号「Java艺术」,可以通过以下二维码关注。转载本文请联系Java艺术公众号


本文转载自网络,原文链接:https://mp.weixin.qq.com/s/DHhGiI74SXZof5JGx_rVRA
本站部分内容转载于网络,版权归原作者所有,转载之目的在于传播更多优秀技术内容,如有侵权请联系QQ/微信:153890879删除,谢谢!

推荐图文

  • 周排行
  • 月排行
  • 总排行

随机推荐