1.概述
- ==: 該操作符生成的是一個boolean結(jié)果,它計算的是操作數(shù)的值之間的關(guān)系
- equals: Object的實例方法,比較兩個對象的content是否相同
- hashCode: Object的native方法,獲取對象的哈希值,用于確定該對象在哈希表中的索引位置,它實際上是一個int類型整數(shù)
2.關(guān)系操作符==
2.1基本數(shù)據(jù)類型變量
在Java中有八種基本數(shù)據(jù)類型:
- 浮點型: float(4 byte), double(8 byte)
- 整型: byte(1 byte), short(2 byte), int(4 byte), long(8 byte)
- 字符型: char(2 byte)
- 布爾型: boolean(JVM規(guī)范沒有明確規(guī)定其所占的空間大小,僅規(guī)定其只能夠取字面值“true”和“false”)
對于這八種基本數(shù)據(jù)類型的變量,變量直接存儲的是“值”。因此,在使用關(guān)系操作符 == 來進行比較時,比較的就是“值”本身。要注意的是,浮點型和整型都是有符號類型的(最高位僅用于表示正負,不參與計算【以 byte 為例,其范圍為 -2^7 ~ 2^7 - 1,-0即-128】),而char是無符號類型的(所有位均參與計算,所以char類型取值范圍為0~2^16-1)。
2.2引用類型變量
在Java中,引用類型的變量存儲的并不是“值”本身,而是與其關(guān)聯(lián)的對象在內(nèi)存中的地址。比如下面這行代碼:
String str;
這句話聲明了一個引用類型的變量,此時它并沒有和任何對象關(guān)聯(lián)。 而通過 new 來產(chǎn)生一個對象,并將這個對象和str進行綁定:
str = new String("HelloWorld");
那么 str1就指向了這個對象,此時引用變量str中存儲的是它指向的對象在內(nèi)存中的存儲地址,并不是“值”本身,也就是說并不是直接存儲的字符串”HelloWorld”。這里面的引用和 C/C++ 中的指針很類似。
2.2小結(jié)
因此,對于關(guān)系操作符 ==:
- 若操作數(shù)的類型是基本數(shù)據(jù)類型,則該關(guān)系操作符判斷的是左右兩邊操作數(shù)的值是否相等
- 若操作數(shù)的類型是引用數(shù)據(jù)類型,則該關(guān)系操作符判斷的是左右兩邊操作數(shù)的內(nèi)存地址是否相同。也就是說,若此時返回true,則該操作符作用的一定是同一個對象。
3.equals
看看equals()方法在Object類中的定義:
public boolean equals(Object obj){
return (this == obj);
}
在String、Double等封裝類中,已經(jīng)重載(overriding)了Object類的equals()方法,于是有了另一種計算公式,是進行內(nèi)容的比較。
比如在String類中:
public int hashCode() {
int h = hash;
if (h == 0) {
char val[] = value;
int len = count;
for (int i = 0; i < len; i++) {
h = 31*h + val[off++];
}
hash = h;
}
return h;
}
4.hashCode
在Object類中的定義為:
public native int hashCode();
是一個本地方法,返回的對象的地址值。
但是,同樣的思路,在String等封裝類中對此方法進行了重寫。方法調(diào)用得到一個計算公式得到的 int值.
5.hashCode和equlas的關(guān)系
- 1、如果兩個對象相同(即用equals比較返回true),那么它們的hashCode值一定要相同;
- 2、如果兩個對象的hashCode相同,它們并不一定相同(即用equals比較返回false)
- 原因:從散列的角度考慮,不同的對象計算哈希碼的時候,可能引起沖突,大家一定還記得數(shù)據(jù)結(jié)構(gòu)中。