在 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 比較的是值是否相等。
掃描下方二維碼,關注更多動態:
系列文章推薦: