原:https://juejin.im/post/5c7f3cb25188251b883cada2
int==Integer為什么返回true
先看現象吧
執行下面的代碼及輸出結果:
int a = 1;
Integer b = 1;
Integer c = new Integer(1);
System.out.println(a==b);//true
System.out.println(a==c);//true
System.out.println(b==c);//false
通常大家對此的解釋是,對於基本類型來說比較的是值,對於引用類型來說比較的是引用,即指向的對象的內存地址。這樣解釋沒錯,bc結果為false毋庸置疑,因為兩個都是引用類型。但是為什么ab(ac)一個是基本類型一個是引用類型,比較的時候還是值比較呢?
這個時候我們不妨把.java源文件編譯后的.class文件使用反編譯工具反編譯成源碼,看看虛擬機內部是如何處理a==b的。
.class文件使用jd-gui反編譯后的:
int a = 1;
Integer b = Integer.valueOf(1);
Integer c = new Integer(1);
System.out.println(a == b.intValue());
System.out.println(a == c.intValue());
System.out.println(b == c);
看到這想必大家都明白了吧,其實基本類型a和引用類型b比較時,引用類型b調用自身的intValue()方法獲取Integer實際表示的int類型的值,即a == b.intValue()還是兩個int類型的變量進行值比較。符合上述:==對於基本類型來說比較的是值,對於引用類型來說比較的是引用,即指向的對象的內存地址。
基本類型及引用類型在內存中的存儲方式
說到這,還要解釋下為什么兩個引用類型的值一樣而引用不一樣以及基本變量為什么是值比較。
其實基本變量int a在內存里只有一份,保存在棧(保存基本類型的變量數據及引用類型的引用)中,Integer b和Integer c中的int值都指向棧中同一個int,不會重新在棧中創建相同的int值。
而對於Integer b和Integer c,其實例是保存在堆(保存所有new出來的對象)中,雖然表示的int值相同,但是在堆中有兩份,每次new都會在堆中開辟一片空間保存new的內容,故Integer b和Integer c分別在兩片不同的內存空間存儲,所以指向的內存地址不同。
[]
而對於Integer b = 1;其反編譯后為Integer b = Integer.valueOf(1);
而valueOf()方法內部是調用了new。
JDK中Integer.valueOf()源碼:
public static Integer valueOf(int i) {
if (i >= IntegerCache.low && i <= IntegerCache.high)
return IntegerCache.cache[i + (-IntegerCache.low)];
return new Integer(i);
}
總結
了解原理是弄清問題的關鍵,像這樣的問題以后還多着呢。不懂的時候可以敲敲代碼,然后反編譯,看看虛擬機是怎么處理的,看看原理翻翻源碼,問題也就迎刃而解了。