廢話很少說了,開門見山吧,先來看一段代碼:java
String str1 = new String("str");數據庫
String str2 = new String("str");this
System.out.println("==比較 :"+ (str1 == str2));hibernate
System.out.println("equal比較:"+ str1.equals(str2));指針
String str3 = "str1";code
String str4 = "str1";對象
System.out.println("==比較 :"+ (str3 == str4));繼承
System.out.println("equal比較:"+ str3.equals(str4));源碼
falsehash
equal比較:true
true
equal比較:true
根據打印的能夠發現使用equal比較時不管是使用自動裝箱來實例化仍是用new來實例化,返回的都true,而用==則不同了,自動裝箱來實例化的返回的是true,而用new來
實例化的返回的確實false;先不急着解決爲何,先來了解下equals和==的區別,到時候就能夠知道答案了
equals方法最初是在全部類的基類Object中進行定義的,源碼是
public boolean equals(Object obj) { return (this == obj); }
能夠看出這裏定義的equals與==是等效的,但上面的怎麼還會不同呢?
緣由就是String類對equals進行了重寫:
public boolean equals(Object anObject) {
if (this == anObject) {
return true;
}
if (anObject instanceof String) {
String anotherString = (String)anObject;
int n = count;
if (n == anotherString.count) {
char v1[] = value;
char v2[] = anotherString.value;
int i = offset;
int j = anotherString.offset;
while (n-- != 0) {
if (v1[i++] != v2[j++])
return false;
}
return true;
}
}
return false;
}
這裏對equals從新須要注意五點:
1 自反性:對任意引用值X,x.equals(x)的返回值必定爲true.
2 對稱性:對於任何引用值x,y,當且僅當y.equals(x)返回值爲true時,x.equals(y)的返回值必定爲true;
3 傳遞性:若是x.equals(y)=true, y.equals(z)=true,則x.equals(z)=true
4 一致性:若是參與比較的對象沒任何改變,則對象比較的結果也不該該有任何改變
5 非空性:任何非空的引用值X,x.equals(null)的返回值必定爲false
通過重寫後就跟==有本質的區別了:
equal:是用來比較兩個對象內部的內容是否相等的,因爲全部的類都是繼承自java.lang.Object類的,因此若是沒有對該方法進行覆蓋的話,調用
的仍然是Object類中的方法,而Object中的equal方法返回的倒是==的判斷,所以,若是在沒有進行該方法的覆蓋後,調用該方法是沒有
任何意義的。在java面向對象的處理中咱們通常在javabean中都要選擇重寫equals方法,使用hibernate後,咱們要生成數據庫的映射文件與實體
類,這是咱們就最好在實體類中進行equals方法的重寫,重寫時咱們能夠根據本身的定義來實現該方法只要遵照那五條原則,例如對於一個student類
咱們定義只要在學號相同時咱們就認爲這兩個對象時相等的;同時咱們還要重寫hashcode方法 ==:是用來判斷兩個對象的地址是否相同,便是否是指相同一個對象。比較的是真正意義上的指針操做。