今天在分析当Integer类型的数据超过-128和127的区间范围时,两个对象的地址比较就是false。通过对其源码的分析后,还想通过反编译.class字节码的方式去分析其原理。
因为是第一次去反编译代码,所有百度去搜了一下idea的反编译插件。
大概说的有两种插件。一个是java bytecode decompile,另一个是Idea jad插件。
其中java bytecode decompile 在我的idea编辑器中是默认安装好的,如果你们没有安装,下载之后记得勾选右边的方框,表示启用插件,apply之后再重启idea就可以了。
因为我的idea版本和最近的插件版本不一致,在反编译时出现了如下错误。
两种解决方法,更新idea版本或者降低插件版本。
之后我们就可以选择项目的.class文件进行decompile。下图是文件的源代码。
可以看到结果是true或false。然后我们去看看他的源代码。
当我们声明一个integer类型的数据时,会根据值的大小判断是从缓存中去取,还是new 一个新的对象。现在我们看看反编译的代码。
C:\Users\Administrator\IdeaProjects\test\out>cd production
C:\Users\Administrator\IdeaProjects\test\out\production>cd test
C:\Users\Administrator\IdeaProjects\test\out\production\test>javap -c IntegerTest
Compiled from “IntegerTest.java”
public class IntegerTest {
public IntegerTest();
Code:
0: aload_0
1: invokespecial #1 // Method java/lang/Object.""?)V
4: return
public static void main(java.lang.String[]);
Code:
0: bipush 100
2: invokestatic #2 // Method java/lang/Integer.valueOf:(I)Ljava/lang/Integer;
5: astore_1
6: bipush 100
8: invokestatic #2 // Method java/lang/Integer.valueOf:(I)Ljava/lang/Integer;
11: astore_2
12: sipush 150
15: invokestatic #2 // Method java/lang/Integer.valueOf:(I)Ljava/lang/Integer;
18: astore_3
19: sipush 150
22: invokestatic #2 // Method java/lang/Integer.valueOf:(I)Ljava/lang/Integer;
25: astore 4
27: getstatic #3 // Field java/lang/System.out:Ljava/io/PrintStream;
30: aload_1
31: aload_2
32: if_acmpne 39
35: iconst_1
36: goto 40
39: iconst_0
40: invokevirtual #4 // Method java/io/PrintStream.println:(Z)V
43: getstatic #3 // Field java/lang/System.out:Ljava/io/PrintStream;
在这里可以注意到的是当我们声明一个Integer的对象时,它是通过valueOf去实现的。这和源码分析得出的结果过是一致的。所以有时通过反编译可能对一个知识点的原理我们能了解的更加深刻。
ps:值得注意的是如果你是从硬盘上安装插件需要把插件放到idea安装目录下plugins目录下。