為什么說兩個 Integer 數值之間不建議使用 “==” 進行比較
本文轉載自微信公眾號「Java極客技術」,作者鴨血粉絲Tang。轉載本文請聯系Java極客技術公眾號。
眾所周知阿里巴巴開發手冊里面有一條強制的規則,說的是在包裝類對象之間的值比較的時候需要使用 equals 方法,在 -128 和 127 之間的數值比較可以使用 ==,如下圖所示。具體的原因相信大家都知道,雖然規則中提到 -128 和 127 之間的數值比較可以使用 ==,但是阿粉強烈建議你還是不要這樣,包裝類統一使用 equals,特別是如果有些數值是通過 API 或者 RPC 接口過來的,一定要注意。
我們看看下面的程序
- public class IntegerEqualTest {
- public static void main(String[] args) {
- Integer a = genA();
- //Integer a = genB();
- Integer b = 0;
- if (a == b) {
- System.out.println("a == 0");
- } else {
- System.out.println("a != 0");
- }
- System.out.println(a == b);
- System.out.println(a == 0);
- }
- private static Integer genA() {
- return new Integer(0);
- }
- private static Integer genB() {
- return 0;
- }
- }
大家可以先看下上面這一段代碼,先猜測一下運行的結果是什么,如果再把 Integer a = genA(); 這行注釋,Integer a = genB(); 這行放開,運行的結果又是什么。
好,1 2 3 結果如下所示
當我們替換注釋那一行的時候,運行結果如下
看到這里其實很多小伙伴都知道是為什么,因為 genA() 方法里面是使用的 Integer 的構造器,構造的是一個新的對象,所以在使用 == 做對比的時候,比較的兩個對象是不一樣的。
是的,原因是這個,但是還有一點沒說清楚那就是為什么在使用 genA() 的時候,下面的結果會不一樣。
- System.out.println(a == b);//false
- System.out.println(a == 0);//true
其實短短的幾行代碼里面,包含了好幾個知識點,分別是自動裝箱拆箱以及 Integer 的 -128 到 127 的數字緩存。
裝箱拆箱
裝箱:自動將基本數據類型轉換為包裝器類型;
拆箱:就是自動將包裝器類型轉換為基本數據類型。
在裝箱的時候自動調用的是 Integer 的 valueOf(int) 方法。而在拆箱的時候自動調用的是 Integer 的 intValue方法。
上面的代碼中 Integer b = 0; 會觸發自動的裝箱調用 Integer valueOf() 方法。而在使用 a == 0 這句的時候,會觸發自動的拆箱。然后我們看源碼會發現有下面緩存的邏輯,其中 IntegerCache.low 是 -128,IntegerCache.high 默認是 127,不過可以通過 JVM 參數進行配置。我們這里的代碼是 0,所以會從緩存中獲取。
- public static Integer valueOf(int i) {
- if (i >= IntegerCache.low && i <= IntegerCache.high)
- return IntegerCache.cache[i + (-IntegerCache.low)];
- return new Integer(i);
- }
為了充分說明 Integer 的緩存,我們看下下面這段程序的執行結果
- Integer c1 = 128;
- Integer c2 = 128;
- System.out.println(c1 == c2);
在運行之前我們先自己分析一下,首先 Integer c1 = 128 和 Integer c2 = 128 按照我們上面說的,會觸發自動裝箱調用 valueOf 方法,通過 valueOf源碼我們可以看到在默認的情況下 128 已經不再 Integer 的緩存里面了,所以 if 條件不滿足會通過 new Integer 構造方法創建兩個對象,所以最終的結果應該是輸出 false。
下面再說一下為什么說在 -128 和 127 以內的也不建議直接使用 == 來實現比較,很顯然就跟我們上面的genA() 方法一樣,很多時候不會一下子就知道一個方法值是怎么得到,即使是緩存范圍以內,別人也有可能是通過構造函數創建出來的,這樣我們在做比較的時候很有可能就會跟預期的不一樣,從而產生事故。
特別是如果通過 RPC 接口獲得返回結果,我們可能連別人的實現方式壓根就看不到,更沒辦法提前知道了。所以我們還是老老實實的按照阿里巴巴的 Java 規范來編寫代碼,采用equals 方法來判斷,這樣肯定沒問題。