你真的知道 == 和 equals 的區別嗎?

王磊的部落格發表於2019-03-07
讓我們每天都有進步,老王帶你打造最全的 Java 面試清單,認真把一件事做到極致。

在 Java 中 == 和 equals 的區別,感覺只有很少的人能才完全說正確。

常見的錯誤回答就是:== 基礎型別對比的是值是否相同,引用型別對比的是引用是否相同;而 equals 則是比較的值是否相同。

至於為什麼說它是錯的,看完本文對 == 和 equals 的解讀,你就知道了。

1、== 解讀

對於基本型別和引用型別 == 的作用效果是不同的,如下所示:

  • 基本型別:比較的是值是否相同;
  • 引用型別:比較的是引用是否相同;

程式碼示例:

String x = "string";
String y = "string";
String z = new String("string");
System.out.println(x==y); // true
System.out.println(x==z); // false
System.out.println(x.equals(y)); // true
System.out.println(x.equals(z)); // true

程式碼解讀:因為 x 和 y 指向的是同一個引用,所以 == 也是 true,而 new String()方法則重寫開闢了記憶體空間,所以 == 結果為 false,而 equals 比較的一直是值,所以結果都為 true。

2、equals 解讀

equals 本質上就是 ==,只不過 String 和 Integer 等重寫了 equals 方法,把它變成了值比較。看下面的程式碼就明白了。

首先來看預設情況下 equals 比較一個有相同值的物件,程式碼如下:

class Cat {
    public Cat(String name) {
        this.name = name;
    }

    private String name;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }
}

Cat c1 = new Cat("王磊");
Cat c2 = new Cat("王磊");
System.out.println(c1.equals(c2)); // false

輸出結果出乎我們的意料,竟然是 false?這是怎麼回事,看了 equals 原始碼就知道了,原始碼如下:

public boolean equals(Object obj) {
        return (this == obj);
}

原來 equals 本質上就是 ==。

那問題來了,兩個相同值的 String 物件,為什麼返回的是 true?程式碼如下:

String s1 = new String("老王");
String s2 = new String("老王");
System.out.println(s1.equals(s2)); // true

同樣的,當我們進入 String 的 equals 方法,找到了答案,程式碼如下:

public boolean equals(Object anObject) {
    if (this == anObject) {
        return true;
    }
    if (anObject instanceof String) {
        String anotherString = (String)anObject;
        int n = value.length;
        if (n == anotherString.value.length) {
            char v1[] = value;
            char v2[] = anotherString.value;
            int i = 0;
            while (n-- != 0) {
                if (v1[i] != v2[i])
                    return false;
                i++;
            }
            return true;
        }
    }
    return false;
}

原來是 String 重寫了 Object 的 equals 方法,把引用比較改成了值比較。

3、總結

總體來說,== 對於基本型別來說是值比較,對於引用型別來說是比較的是引用;而 equals 預設情況下是引用比較,只是很多類重寫了 equals 方法,比如 String、Integer 等把它變成了值比較,所以一般情況下 equals 比較的是值是否相等。

掃描下方二維碼,關注更多動態:

系列文章推薦:

Java 最常見的 200+ 面試題

程式設計師精美簡歷Top榜—面試必備

你真的懂 == 和 equals 的區別嗎?

相關文章