Java中==與equals()及hashcode()三者之間的關(guān)系詳解
1.= =
=為賦值運(yùn)算符,==為比較運(yùn)算符,僅比較對(duì)象的內(nèi)存地址,無法比較真正意義上的相等!
JDK里的equals方法就是通過==來實(shí)現(xiàn)的比較對(duì)象的內(nèi)存地址
以Integer為例
Integer a = 127; Integer b = 127; System.out.println(a == b);//true Integer c = 128; Integer d = 128; System.out.println(c == d);//false
這里也是通過== 引出一個(gè)知識(shí)點(diǎn),一個(gè)數(shù)值之差為啥導(dǎo)致結(jié)果不一樣?在[-128,127]的區(qū)間內(nèi)Integer a = 127;由于設(shè)計(jì)了緩存,而后的Integer b = 127;就是直接利用的緩存里的數(shù)值對(duì)象,所以通過==比較的結(jié)果為true,因?yàn)樗麄儽举|(zhì)還是一個(gè)數(shù)值對(duì)象
而Integer c = 128;Integer d = 128;就沒這樣幸運(yùn)了,超過了緩存區(qū)間會(huì)重新new出對(duì)象,以至于兩者雖然數(shù)值相同但是地址不同
所以啊,==比較的是地址!也只是地址!
在前面==的基礎(chǔ)上,再來看equals()
2.equals()
以一個(gè)String類型的變量為例,當(dāng)我們來使用equals()比較兩個(gè)對(duì)象時(shí),結(jié)果肯定是false,因?yàn)閑quals()方法比較的是內(nèi)存地址,這里的person1,person2均是兩次new出來的,所以地址肯定是不相同的,而person1,person3指向同一空間地址一定是相同的
Person person1 = new Person("lyy");
Person person2 = new Person("lyy");
Person person3 = person1;
System.out.println(person1.equals(person2));//false
System.out.println(person1.equals(person3));//true
在不重寫的情況下,我們Ctrl+B看一下equals()的源碼:
public boolean equals(Object obj) {
return (this == obj);
}
顯而易見的是 (this = = obj)是該方法的核心,而 = = 又是兩個(gè)對(duì)象比較的方式,= = 嘛比的是內(nèi)存地址,懂的都懂噢
3.重寫equals()
記不記得你在刷面經(jīng)或者短視頻的時(shí)候經(jīng)??吹降囊痪湓?mdash;—比較兩個(gè)對(duì)象的內(nèi)容是否相等時(shí)我們要重寫equals()方法!
這是為何?那我們不妨來重寫一下equals()試試水
@Override
public boolean equals(Object obj){
if (this==obj){
return true;
}
if (obj==null||getClass()!= obj.getClass()){
return false;
}
Person person=(Person)obj;
return Objects.equals(name,person.name);
}
重寫過equals后,原有的兩者就已發(fā)生翻天覆地的變化,從原來的比較內(nèi)存地址——>比較對(duì)象內(nèi)容,這是一件很神奇的事情,因?yàn)閷?shí)現(xiàn)了比較不同對(duì)象的相同或者不同內(nèi)容!
重寫之后:

具體是如何實(shí)現(xiàn)的呢?就像下面這樣…
4.equals()比較流程
下面我們來探索一下重寫的equals()是如何比較內(nèi)容的:
通過debug來深入理解一下

下面來看一下debug過程中變量情況

總的來說,通過debug,重寫equals()來比較不同變量的不同或者相同內(nèi)容得到了進(jìn)一步論證!
5.hashcode()
我們?cè)贗DEA中通過CTRL+O的快捷鍵重寫hashcode()時(shí)它上面所屬的類是誰?

java.lang.Object!
顯而易見,該方法是Object類所定義的方法,作用是返回對(duì)象的哈希值返回值的類型為int(哈希值的作用是確定該對(duì)象在哈希表中的位置),曾經(jīng)有這樣一句流川千古的話:你必須在每個(gè)重寫equals()的類中重寫一遍hashcode()方法
如果不這樣做將會(huì)違反Object.hashcode()的一般約定,這會(huì)阻止lei與所有基于散列的集合(比如hashmap,hashset…)一起正常工作。為啥?因?yàn)閔ashmap,hashset等基于散列的集合中,會(huì)使用對(duì)象的hashcode值來確定該對(duì)象應(yīng)該如何存儲(chǔ)到集合中,并且再次使用hashcode來定位對(duì)象在集合中的位置
那么
在一個(gè)類中重寫了equals()但沒重寫hashcode()會(huì)出現(xiàn)啥情況呢?
我們來通過一個(gè)例子試試水~
嘗試把對(duì)象都放入一個(gè)不能重復(fù)的set里,然后看集合的長(zhǎng)度來判斷兩個(gè)對(duì)象是否相等!
public class equals_hashcode {
public static void main(String[] args) {
Person person1 = new Person("lyy");
Person person2 = new Person("lyy");
HashSet<Person> set = new HashSet<>();
set.add(person1);
set.add(person2);
System.out.println(set.size());
}
}
class Person {
public String name;
public Person(String name) {
this.name = name;
}
@Override
public boolean equals(Object obj) {
if (this == obj) {
return true;
}
if (obj == null || getClass() != obj.getClass()) {
return false;
}
Person person = (Person) obj;
return Objects.equals(name, person.name);
}
@Override
public int hashCode() {
return Objects.hash(name);
}
}情況一:
當(dāng)只重寫了equals()方法時(shí)運(yùn)行的結(jié)果為2,由此得出兩個(gè)對(duì)象不相等!

情況二:
當(dāng)既重寫了equals()和hashcode()后運(yùn)行結(jié)果為1,所以兩個(gè)對(duì)象相等!

由此得出,對(duì)象相等的本質(zhì)是:
1.地址相同
2.哈希值相同(重寫hashcode的體現(xiàn))
這也不難聯(lián)系到之前的約定了,在比較對(duì)象是否相等的場(chǎng)景下,我們必須重寫equals()和hashcode()!
對(duì)象相等建立在==之上,equals(),hashcode()的雙重寫是對(duì)象相等的基本準(zhǔn)則!
到此這篇關(guān)于Java中==與equals()及hashcode()三者之間的關(guān)系詳解的文章就介紹到這了,更多相關(guān)Java == equals() hashcode()內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- Java ==,equals()與hashcode()的使用
- 詳解Java中==和equals()的區(qū)別
- java中“==“和equals()的區(qū)別詳解
- 詳解java==運(yùn)算符和equals()方法的區(qū)別
- Java中比較運(yùn)算符compareTo()、equals()與==的區(qū)別及應(yīng)用總結(jié)
- Java中==符號(hào)與equals()的使用詳解(測(cè)試兩個(gè)變量是否相等)
- Java中的== 和equals()方法詳解與實(shí)例
- 詳解Java中“==”與equals()的區(qū)別
- Java中的==和equals()區(qū)別小結(jié)
相關(guān)文章
SpringSceurity實(shí)現(xiàn)短信驗(yàn)證碼登陸
這篇文章主要介紹了SpringSceurity實(shí)現(xiàn)短信驗(yàn)證碼登陸,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-06-06
java實(shí)現(xiàn)手寫一個(gè)簡(jiǎn)單版的線程池
有些人可能對(duì)線程池比較陌生,并且更不熟悉線程池的工作原理。本文就來手寫一個(gè)簡(jiǎn)單版的線程池,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-08-08
java實(shí)現(xiàn)頁(yè)面多查詢條件必選的統(tǒng)一處理思路
這篇文章主要為大家介紹了java實(shí)現(xiàn)頁(yè)面多查詢條件必選的統(tǒng)一處理思路詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-06-06
java后端如何實(shí)現(xiàn)防止接口重復(fù)提交
這篇文章主要介紹了java后端如何實(shí)現(xiàn)防止接口重復(fù)提交問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-05-05
Java編程使用UDP建立群聊系統(tǒng)代碼實(shí)例
這篇文章主要介紹了Java編程使用UDP建立群聊系統(tǒng)代碼實(shí)例,具有一定借鑒價(jià)值,需要的朋友可以參考下。2018-01-01
JWT在OpenFeign調(diào)用中進(jìn)行令牌中繼詳解
Feign是一個(gè)聲明式的Web Service客戶端,是一種聲明式、模板化的HTTP客戶端。而OpenFeign是Spring Cloud 在Feign的基礎(chǔ)上支持了Spring MVC的注解,如@RequesMapping等等,這篇文章主要給大家介紹了關(guān)于JWT在OpenFeign調(diào)用中進(jìn)行令牌中繼的相關(guān)資料,需要的朋友可以參考下2021-10-10
Java實(shí)現(xiàn)京東聯(lián)盟API數(shù)據(jù)獲取功能
這篇文章介紹了Java獲取京東聯(lián)盟API數(shù)據(jù)的方法,文中通過示例代碼介紹的非常詳細(xì)。對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-07-07
基于Java中throw和throws的區(qū)別(詳解)
下面小編就為大家?guī)硪黄贘ava中throw和throws的區(qū)別(詳解)。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2017-07-07

