欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

淺談java中為什么重寫equals后需要重寫hashCode

 更新時間:2021年05月23日 16:16:30   作者:阿布QAQ~  
今天帶各位學(xué)習(xí)一下java中為什么重寫equals后需要重寫hashCode,文中有非常詳細的圖文介紹及代碼示例,對正在學(xué)習(xí)java的小伙伴們有很好的幫助,需要的朋友可以參考下

一、先看現(xiàn)象

public class TestDemo {

    public static void main(String[] args) {
        Person p1 = new Person("阿倫");
        Person p2 = new Person("阿倫");
        System.out.println(p1.equals(p2));
    }

    static class Person {

        public Person(String name) {
            this.name = name;
        }
        private String name;
        public String getName() {
            return name;
        }
        public void setName(String name) {
            this.name = name;
        }
        @Override
        public boolean equals(Object o) {
            if (this == o) return true;
            if (o == null || getClass() != o.getClass()) return false;
            Person person = (Person) o;
            return Objects.equals(name, person.name);
        }
    }

}

上方代碼運行結(jié)束后,可以看到輸出的結(jié)果為true。

image.png

可以看到,因為重寫了equals方法后,只要類型相同,name相同,就認定兩個對象是同一個,而不是默認用內(nèi)存地址去比對是不是同一個,

在我們的業(yè)務(wù)代碼中一般會需要這么用,例如一個人名字相等,就是同一個人,正常來說,我們重寫了equals判斷是相等的不就已經(jīng)夠用了嗎,為什么還要重寫hashCode呢, 下面我們再看一個例子。

二、為什么要重寫hashCode

根據(jù)開篇中的代碼,我們不重寫hashCode,下面來做一個需求,過濾重復(fù)的人,例如兩個名叫阿倫的人, 但是我們只保存一個,

那這時候咱們用HashMap來做,因為HashMapkey是唯一的,去重的,我們重寫了Personequals方法,只要名字是相等的,就是同一個對象,那HashMap應(yīng)該會識別出來并去重吧?我們看下結(jié)果:

public static void main(String[] args) {
        Person p1 = new Person("阿倫");
        Person p2 = new Person("阿倫");
        Map<Person, String> map = new HashMap<>();
        map.put(p1, p1.getName());
        map.put(p2, p2.getName());
        System.out.println("map長度:" + map.size());
        map.forEach((key, value) -> {
            System.out.println(key.getName());
        });
    }

image.png

是不是很神奇,竟然沒有去重,兩個都保存到了HashMap中,并沒有達到我們想要的效果,這是為什么?

首先HashMap的內(nèi)部存值是一個數(shù)組,但是HashMap的查找速度非??欤臼荗(1),

這是因為它借助了Hash算法,HashMapkey進行hash后,得出一個整數(shù)值, 這個整數(shù)值就是存放到最終的存儲數(shù)組中的下標, 當然這塊后續(xù)還有一系列的處理, 可以查閱相關(guān)資料做深入的了解, 這里只做簡單的描述,

我們接著上面的問題看,因為我們沒有重寫hashCode方法,雖然equals以兩個對象的name值是否相同做對比,但是HashMap存值的時候,是通過hashCode進行計算,算出一個值存到相應(yīng)的數(shù)組下標下去的呀?

所以我們上面代碼中的p1p2兩個值返回的hashCode值是不同的,所以計算出來的下標也不同,導(dǎo)致他們被HashMap存到不同的數(shù)組下標下面去了~ 這就是為什么沒有去重成功的原因,

image.png

看上圖, 兩個對象在堆地址中, 名字是一樣的,hashCode不同,如果是通過名字做比對,做hash,那么就是相等的,但是HashMap用的是hashCode,所以,我們需要重寫hashCode方法,根據(jù)自身業(yè)務(wù)保證,相同含義在業(yè)務(wù)層面屬于一個對象的hashCode也要保持一致。

看下圖,我們可以將hashCode的計算方式改成用name來計算:

image.png

三、實現(xiàn)代碼

public class TestDemo {

    public static void main(String[] args) {
        Person p1 = new Person("阿倫");
        Person p2 = new Person("阿倫");
        System.out.println(p1.hashCode());
        System.out.println(p2.hashCode());
        Map<Person, String> map = new HashMap<>();
        map.put(p1, p1.getName());
        map.put(p2, p2.getName());
        map.get(p1);
        System.out.println("map長度:" + map.size());
        map.forEach((key, value) -> {
            System.out.println(key.getName());
        });
    }

    static class Person {

        public Person(String name) {
            this.name = name;
        }
        private String name;
        public String getName() {
            return name;
        }
        public void setName(String name) {
            this.name = name;
        }
        @Override
        public boolean equals(Object o) {
            if (this == o) return true;
            if (o == null || getClass() != o.getClass()) return false;
            Person person = (Person) o;
            return Objects.equals(name, person.name);
        }
        @Override
        public int hashCode() {
            return Objects.hash(name);
        }
    }
}

image.png

可以看到,當我們重寫了hashCode讓對象的名字作為計算的值,用來產(chǎn)生最終的hash值,這樣HashMap就可以幫我們把兩個對象,路由到一個下標下面了,再通過equals比對,確定兩個是同一個對象,從而達到去重的效果。

四、總結(jié)

根據(jù)業(yè)務(wù)狀況重寫equals后,一定要將hashCode用一定相同的規(guī)則做hash,防止在一些需要用到對象hashCode的地方造成誤會,引發(fā)問題,

同時這里再說一點,hashCode也會發(fā)生沖突和重復(fù)喔~ 也許他們并不是一個對象,但是hashCode是相同的,這時HashMap是怎么處理的呢?

這里簡單說一下,是用鏈表,當兩個對象hashCode沖突時,會將這兩個對象放在同一個下標下的鏈表中都保存著,獲取的時候通過hashCode路由到相應(yīng)的地點,然后循環(huán)這個列表通過equals方法做對比,返回最終的正確值。

重寫equalshashCode的原則:

1.自反性:x.equals(x) == true,自己和自己比較相等

2.對稱性:x.equals(y) == y.equals(x),兩個對象調(diào)用equals的的結(jié)果應(yīng)該一樣

3.傳遞性:如果x.equals(y) == true y.equals(z) == true 則 x.equals(z) == true,x和y相等,y和z相等,則x和z相等

4.一致性 : 如果x對象和y對象有成員變量num1和num2,其中重寫的equals方法只有num1參加了運算,則修改num2不影響x.equals(y)的值

到此這篇關(guān)于淺談java中為什么重寫equals后需要重寫hashCode的文章就介紹到這了,更多相關(guān)Java重寫equals內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • SpringBoot中的共享Session域詳解

    SpringBoot中的共享Session域詳解

    這篇文章主要介紹了SpringBoot中的共享Session域詳解,使用Redis解決Session共享問題的原理非常簡單,就是把原本存儲在不同服務(wù)器上的Session拿出來放在一個獨立的服務(wù)器上,需要的朋友可以參考下
    2024-01-01
  • Java?agent的入門與demo演示詳解

    Java?agent的入門與demo演示詳解

    Java?Agent?是?Java?1.5?版本之后引?的特性,其主要作?是在class被加載之前對其攔截,這篇文章主要介紹了agent的簡單使用,需要的可以參考下
    2023-05-05
  • springboot之自動裝配全過程

    springboot之自動裝配全過程

    這篇文章主要介紹了springboot之自動裝配全過程,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教
    2024-08-08
  • 淺談Java基準性能測試之JMH

    淺談Java基準性能測試之JMH

    JMH是Java Microbenchmark Harness的簡稱,一個針對Java做基準測試的工具。想準確的對一段代碼做基準性能測試并不容易,因為JVM層面在編譯期、運行時對代碼做很多優(yōu)化,當代碼塊處于整個系統(tǒng)中運行時并不一定會生效,產(chǎn)生錯誤的基準測試結(jié)果,這個問題就是JMH要解決的
    2021-06-06
  • java解析json復(fù)雜數(shù)據(jù)的方法詳解

    java解析json復(fù)雜數(shù)據(jù)的方法詳解

    這篇文章主要為大家詳細介紹了java解析json復(fù)雜數(shù)據(jù)的兩種常用方法,文中的示例代碼講解詳細,具有一定的借鑒價值,需要的小伙伴可以了解下
    2024-01-01
  • mybatis插入與批量插入返回ID的原理詳解

    mybatis插入與批量插入返回ID的原理詳解

    這篇文章主要給大家介紹了關(guān)于mybatis插入與批量插入返回ID的原理的相關(guān)資料,文中通過示例代碼介紹的非常詳細,對大家學(xué)習(xí)或者使用mybatis具有一定的參考學(xué)習(xí)價值,需要的朋友們下面來一起學(xué)習(xí)學(xué)習(xí)吧
    2019-07-07
  • Spring Boot 統(tǒng)一數(shù)據(jù)返回格式的解決方案

    Spring Boot 統(tǒng)一數(shù)據(jù)返回格式的解決方案

    統(tǒng)?的數(shù)據(jù)返回格式使? @ControllerAdvice 和 ResponseBodyAdvice 的?式實現(xiàn),下面給大家分享Spring Boot 統(tǒng)一數(shù)據(jù)返回格式的解決方案,感興趣的朋友一起看看吧
    2024-03-03
  • Java之while與do-while循環(huán)的用法詳解

    Java之while與do-while循環(huán)的用法詳解

    在上一篇文章中,給大家講解了循環(huán)的概念,并重點給大家講解了for循環(huán)的使用。但在Java中,除了for循環(huán)之外,還有while、do-while、foreach等循環(huán)形式。這篇文章給大家講解while循環(huán)的使用
    2023-05-05
  • Netty啟動流程服務(wù)端channel初始化源碼分析

    Netty啟動流程服務(wù)端channel初始化源碼分析

    這篇文章主要為大家介紹了Netty啟動流程服務(wù)端channel初始化源碼分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2022-03-03
  • MyBatis查詢數(shù)據(jù)庫語句總結(jié)

    MyBatis查詢數(shù)據(jù)庫語句總結(jié)

    MyBatis是一種持久化框架,可以與許多不同類型的關(guān)系型數(shù)據(jù)庫連接,下面這篇文章主要給大家介紹了關(guān)于MyBatis查詢數(shù)據(jù)庫語句的相關(guān)資料,文中通過實例代碼介紹的非常詳細,需要的朋友可以參考下
    2023-06-06

最新評論