漫畫:Integer 竟然有 4 種比較方法?
本文轉(zhuǎn)載自微信公眾號「Java中文社群」,作者磊哥 。轉(zhuǎn)載本文請聯(lián)系 Java中文社群公眾號。
代碼測試
- public class IntegerTest {
- public static void main(String[] args) {
- Integer i1 = 127;
- Integer i2 = 127;
- System.out.println(i1 == i2);
- Integer i3 = 128;
- Integer i4 = 128;
- System.out.println(i3 == i4);
- }
- }
以上代碼的執(zhí)行結(jié)果為:
- true
- false
首先,當(dāng)我們將以上的測試代碼編譯為字節(jié)碼(.class)之后,編碼的代碼如下:
- public class IntegerTest {
- public static void main(String[] paramArrayOfString) {
- Integer integer1 = Integer.valueOf(127);
- Integer integer2 = Integer.valueOf(127);
- System.out.println((integer1 == integer2));
- Integer integer3 = Integer.valueOf(128);
- Integer integer4 = Integer.valueOf(128);
- System.out.println((integer3 == integer4));
- }
- }
可以看出在創(chuàng)建 Integer 時使用到了 valueOf,它的實現(xiàn)源碼如下:
- public static Integer valueOf(int i) {
- if (i >= IntegerCache.low && i <= IntegerCache.high)
- return IntegerCache.cache[i + (-IntegerCache.low)];
- return new Integer(i);
- }
從上述源碼中可以看出這個方法中使用了 IntegerCache,IntegerCache 的源碼如下:
- private static class IntegerCache {
- static final int low = -128;
- static final int high;
- static final Integer cache[];
- static {
- // high value may be configured by property
- int h = 127;
- String integerCacheHighPropValue =
- sun.misc.VM.getSavedProperty("java.lang.Integer.IntegerCache.high");
- if (integerCacheHighPropValue != null) {
- try {
- int i = parseInt(integerCacheHighPropValue);
- i = Math.max(i, 127);
- // Maximum array size is Integer.MAX_VALUE
- h = Math.min(i, Integer.MAX_VALUE - (-low) -1);
- } catch( NumberFormatException nfe) {
- // If the property cannot be parsed into an int, ignore it.
- }
- }
- high = h;
- cache = new Integer[(high - low) + 1];
- int j = low;
- for(int k = 0; k < cache.length; k++)
- cache[k] = new Integer(j++);
- // range [-128, 127] must be interned (JLS7 5.1.7)
- assert IntegerCache.high >= 127;
- }
- private IntegerCache() {}
- }
從上述源碼可以看出,在 Integer 的取值在 -128 到 127 之間時,它會復(fù)用已有的對象,因此在 i1(127)和 i2 使用 == 對比時值才會為 true,而當(dāng)取值變?yōu)?128 時,則執(zhí)行的結(jié)果為 false。
這一點其實在阿里巴巴的《Java開發(fā)手冊》中也有相應(yīng)的規(guī)定,規(guī)定的內(nèi)容如下:
【強制】所有整型包裝類對象之間值的比較,全部使用 equals 方法比較。
說明:對于 Integer var = ? 在 -128 至 127 之間的賦值,Integer 對象是在 IntegerCache.cache 產(chǎn)生, 會復(fù)用已有對象,這個區(qū)間內(nèi)的 Integer 值可以直接使用 == 進行判斷,但是這個區(qū)間之外的所有數(shù)據(jù),都 會在堆上產(chǎn)生,并不會復(fù)用已有對象,這是一個大坑,推薦使用 equals 方法進行判斷。
注意事項
不僅如此,當(dāng)我們使用 new Integer 時,無論值為多少都不能使用 == 比較,示例代碼如下:
- public class IntegerTest {
- public static void main(String[] args) {
- Integer i1 = new Integer(127);
- Integer i2 = new Integer(127);
- System.out.println(i1 == i2);
- }
- }
以上代碼的執(zhí)行結(jié)果為:
- false
這是因為 new Integer 方法并沒有使用到 IntegerCache,而是直接創(chuàng)建了新對象,因此就不能用 == 比較了。
小貼士:== 是用來直接比對兩個對象的引用是否相同的,而 equals 則是用來對比兩個對象的值是否相同的。
其他比較方式
compareTo
因為 Integer 類實現(xiàn)了 Comparable 接口,因此我們可以使用 compareTo 來對比兩個值的大小,實現(xiàn)源碼如下:
- public final class Integer extends Number implements Comparable<Integer> {
- // 忽略其他內(nèi)容
- }
compareTo 的使用如下:
- public class IntegerTest {
- public static void main(String[] args) {
- Integer i1 = new Integer(128);
- Integer i2 = new Integer(128);
- System.out.println(i1.compareTo(i2));
- }
- }
以上代碼的執(zhí)行結(jié)果為:
- 0
compareTo 的源碼如下:
- public int compareTo(Integer anotherInteger) {
- return compare(this.value, anotherInteger.value);
- }
- public static int compare(int x, int y) {
- return (x < y) ? -1 : ((x == y) ? 0 : 1);
- }
由此可以看出 compareTo 的返回值總共有三個:-1、0、1,其中 -1 表示前一值小于后一個值;0 表示兩個值相等;1 表示前一個值大于后一個值,因此我們用它來比較兩個 Integer 的值是否相等。
直接運算
compareTo 方法給我們了一個啟發(fā),我們可以直接將兩個值進行相減,如果相減的值等于 0,則說明對比的兩個值是相同的,示例代碼如下:
- public class IntegerTest {
- public static void main(String[] args) {
- Integer i1 = new Integer(128);
- Integer i2 = new Integer(128);
- System.out.println((i1 - i2) == 0);
- }
- }
以上代碼的執(zhí)行結(jié)果為:
- true
擴展知識:IntegerCache 值域修改
IntegerCache 默認的取值范圍為 -128 到 127,但我們可以通過設(shè)置啟動參數(shù)來調(diào)整 IntegerCache 的最大緩存值,比如我們可以配置虛擬機的啟動參數(shù) -XX:AutoBoxCacheMax=1000,此配置表示將緩存的最大值設(shè)置為 1000,如果是 Idea 的配置如下:

此時我們編寫一個測試代碼:
- public class IntegerTest {
- public static void main(String[] args) {
- Integer i1 = 999;
- Integer i2 = 999;
- System.out.println(i1 == i2);
- }
- }
以上代碼的執(zhí)行結(jié)果為:
- true
從運行的結(jié)果可以看出 IntegerCache 的取值范圍被成功的更改了。
總結(jié)
本文我們介紹了 Integer 的四種比較方式:==、equals、compareTo、直接運算,而 == 方式并不能用于 Integer 的比較,它只適用于非 new Integer 的一定范圍內(nèi)(-128~127),而后三種方式都可以正常用于 Integer 的比較,其中 equals 的比較方式是最簡單也是最通用的。
原文鏈接:https://mp.weixin.qq.com/s/b9D_iaxX6SrNrw3WO55nVw