為什么 Java 中“1000==1000”為false,而”100==100“為true?
2022-04-05 22:01:32 來源:程序員客棧
這是一個挺有意思的討論話題。
如果你運行下面的代碼:
Integer?a?=?1000,?b?=?1000;??????System.out.println(a?==?b);//1????Integer?c?=?100,?d?=?100;??????System.out.println(c?==?d);//2
你會得到:
falsetrue
基本知識:我們知道,如果兩個引用指向同一個對象,用==表示它們是相等的。如果兩個引用指向不同的對象,用==表示它們是不相等的,即使它們的內容相同。
因此,后面一條語句也應該是false 。
這就是它有趣的地方了。如果你看去看 Integer.java 類,你會發現有一個內部私有類,IntegerCache.java,它緩存了從-128到127之間的所有的整數對象。
所以事情就成了,所有的小整數在內部緩存,然后當我們聲明類似——
Integer?c?=?100;
的時候,它實際上在內部做的是
Integer?i?=?Integer.valueOf(100);
現在,如果我們去看valueOf()方法,我們可以看到:
public?static?Integer?valueOf(int?i)?{??????if?(i?>=?IntegerCache.low?&&?i??????????return?IntegerCache.cache[i?+?(-IntegerCache.low)];??????return?new?Integer(i);????}
如果值的范圍在-128到127之間,它就從高速緩存返回實例。
所以…
Integer?c?=?100,?d?=?100;
指向了同一個對象。
這就是為什么我們寫
System.out.println(c?==?d);
我們可以得到true。
現在你可能會問,為什么這里需要緩存?
合乎邏輯的理由是,在此范圍內的“小”整數使用率比大整數要高,因此,使用相同的底層對象是有價值的,可以減少潛在的內存占用。
然而,通過反射API你會誤用此功能。
運行下面的代碼,享受它的魅力吧!
public?static?void?main(String[]?args)?throws?NoSuchFieldException,?IllegalAccessException?{???????Class?cache?=?Integer.class.getDeclaredClasses()[0];?//1??????Field?myCache?=?cache.getDeclaredField("cache");?//2??????myCache.setAccessible(true);//3???????Integer[]?newCache?=?(Integer[])?myCache.get(cache);?//4??????newCache[132]?=?newCache[133];?//5???????int?a?=?2;??????int?b?=?a?+?a;??????System.out.printf("%d?+?%d?=?%d",?a,?a,?b);?//????}
如果本文對你有幫助的話,請不要吝嗇你的贊,謝謝!
相關閱讀
-
為什么 Java 中“1000==1000”為false...
這是一個挺有意思的討論話題。如果你運行下面的代碼:Integera=1000,... -
去米哈游吧!
看到一個實習offer選擇提問,他糾結于去米哈游還是去字節實習,這是... -
NeurIPS 2021 Spotlight | 準確、快...
點擊上方“邁微AI研習社”,選擇“星標★”公眾號重磅干貨,第一時... -
工業人工智能機器人介紹(二)
怎樣從機器人的“手相”看出它是干什么職業的呢?很簡單。瞧,三鉤... -
蔣爍淼云計算創業十二年:從云廠商的影子...
為什么要講這個故事呢?因為那是一個云廠商影子,走出自我,在突破中... -
漏洞發現:代碼分析引擎 CodeQL
codeql是一門類似SQL的查詢語言,通過對源碼(C C++、C 、golang...