开发过程中遇到如下问题
Long a = 100L; Long b = 100L; System.out.println(a == b); System.out.println(a.equals(b)); System.out.println(a == 100); System.out.println(a.equals(100));
输出结果:
true true true false
但是当Long类型大于127时:
Long a = 128L; Long b = 128L; System.out.println(a == b); System.out.println(a.equals(b)); System.out.println(a == 128); System.out.println(a.equals(128));
输出结果:
false true true false
查看源码:java.lang.Long.java
LongCache会预先缓存-128–127范围内的数,通过缓存频繁请求的值代来更好的空间和时间性能,
当数据超出此范围,则new一个Long对象;
“==”是比较的地址,超出此范围的数据地址不一致,所以范围内的比较是true,范围外的数据是false;
而a==100则实现了类型的自动向上转换,将int类型转换成Long进行对比,所以输出true;
在Long.java里重写了equals()方法,先进行类型对比,在进行值的对比,所以a.equals(100)输出false;
# 三、源码分析(反汇编法)
我们先看下面的示例代码,并思考该段代码的输出结果:
public class IntTest { public static void main(String[] args) { Integer a = 100, b = 100, c = 150, d = 150; System.out.println(a == b); System.out.println(c == d); } }
通过运行代码可以得到答案,程序输出的结果分别为: true , false。
首先编译源代码:javac IntTest.java
然后需要对代码进行反汇编,执行:javap -c IntTest
反编译后,我们得到以下代码:
Compiled from "IntTest.java" public class com.chujianyun.common.int_test.IntTest { public com.chujianyun.common.int_test.IntTest(); Code: 0: aload_0 1: invokespecial #1 // Method java/lang/Object."<init>":()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; 46: aload_3 47: aload 4 49: if_acmpne 56 52: iconst_1 53: goto 57 56: iconst_0 57: invokevirtual #4 // Method java/io/PrintStream.println:(Z)V 60: return }
可以明确得 "看到" 这四个 `Integer var = ? 形式声明的变量的确是通过 java.lang.Integer#valueOf(int) 来构造 Integer
对象的。
接下来对汇编后的代码进行详细分析,如果看不懂可略过:
根据《Java Virtual Machine Specification : Java SE 8 Edition》3,后缩写为 JVMS , 第 6 章 虚拟机指令集的相关描述以及《深入理解 Java 虚拟机》4 414-149 页的 附录 B “虚拟机字节码指令表”。 我们对上述指令进行解读:
由于该指令有以下特性:
if_acmpeq 比较栈两个引用类型数值,相等则跳转
if_acmpne 比较栈两个引用类型数值,不相等则跳转
可知参数描述符为 Z ,返回值描述符为 V。
根据 4.3.2 字段描述符 ,可知 FieldType 的字符为 Z 表示 boolean 类型, 值为 true 或 false。
根据 4.3.3 字段描述符 ,可知返回值为 void。
因此可以知,最终调用了 java.io.PrintStream#println(boolean) 函数打印栈顶常量即 true。
同样地我们也编写一个Long类型的示例片段:
public class LongTest { public static void main(String[] args) { Long a = -128L, b = -128L, c = 150L, d = 150L; System.out.println(a == b); System.out.println(c == d); } }
得到下面反编译的代码:
public class com.imooc.basic.learn_int.LongTest { public com.imooc.basic.learn_int.LongTest(); Code: 0: aload_0 1: invokespecial #1 // Method java/lang/Object."<init>":()V 4: return public static void main(java.lang.String[]); Code: 0: ldc2_w #2 // long -128l 3: invokestatic #4 // Method java/lang/Long.valueOf:(J)Ljava/lang/Long; 6: astore_1 7: ldc2_w #2 // long -128l 10: invokestatic #4 // Method java/lang/Long.valueOf:(J)Ljava/lang/Long; 13: astore_2 14: ldc2_w #5 // long 150l 17: invokestatic #4 // Method java/lang/Long.valueOf:(J)Ljava/lang/Long; 20: astore_3 21: ldc2_w #5 // long 150l 24: invokestatic #4 // Method java/lang/Long.valueOf:(J)Ljava/lang/Long; 27: astore 4 29: getstatic #7 // Field java/lang/System.out:Ljava/io/PrintStream; 32: aload_1 33: aload_2 34: if_acmpne 41 37: iconst_1 38: goto 42 41: iconst_0 42: invokevirtual #8 // Method java/io/PrintStream.println:(Z)V 45: getstatic #7 // Field java/lang/System.out:Ljava/io/PrintStream; 48: aload_3 49: aload 4 51: if_acmpne 58 54: iconst_1 55: goto 59 58: iconst_0 59: invokevirtual #8 // Method java/io/PrintStream.println:(Z)V 62: return }
我们从上述代码中发现 Long var = ? 的确是通过 java.lang.Long#valueOf(long) 来构造对象的。
对于Long类型的对比,不要用“==”,尽量避免Long类型的直接对比
将Long转换成基本类型再进行比较:a.longValue() == b.longValue(),或者0 == Long.compare(a, b);