開發過程當中遇到以下問題java
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範圍內的數,經過緩存頻繁請求的值代來更好的空間和時間性能,code
當數據超出此範圍,則new一個Long對象;對象
「==」是比較的地址,超出此範圍的數據地址不一致,因此範圍內的比較是true,範圍外的數據是false;ip
而a==100則實現了類型的自動向上轉換,將int類型轉換成Long進行對比,因此輸出true;ci
在Long.java裏重寫了equals()方法,先進行類型對比,在進行值的對比,因此a.equals(100)輸出false;開發
# 3、源碼分析(反彙編法)
咱們先看下面的示例代碼,並思考該段代碼的輸出結果:
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);