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

JAVA hashCode使用方法詳解

 更新時間:2013年11月14日 14:29:20   作者:  
本文詳細解釋了JAVA hashCode的使用方法,提供了測試hashCode和equals方法的使用實例

一.問題引入
談到hashCode就不得不說equals方法,二者均在Object類里,由于Object類是所有類的基類,所以一切類里都可以重寫這兩個方法。
要想較清晰的理解,需要先知道容器Collection,Set,list,Map(key值不可重復),Set元素無序不重復,list元素有序可重復,那么JVM是如何確定不同的元素的呢?
難道是逐個比較么,那樣效率就太低了,JVM采用hash的方法(hash地址不一定是實際的物理地址),看看這個地址上是否有內(nèi)容,沒的話就認為不存在相同對象……
 且看下面分解……

二.問題分析
首先equals()和hashcode()這兩個方法都是從object類中繼承過來的,equals()方法在object類中定義如下:

復制代碼 代碼如下:

public boolean equals(Object obj) {
    return (this == obj);
}

從聲明看出很明顯是對兩個對象的地址值進行的比較(即比較引用是否相同)。但是我們必需清楚,當String 、Math、還有Integer、Double。。。。等這些封裝類在使用equals()方法時,已經(jīng)覆蓋了object類的
equals()方法。

2. 其次是hashcode() 方法,在object類中定義如下:
public native int hashCode();
說明是一個本地方法,它的實現(xiàn)是根據(jù)本地機器相關(guān)的。

復制代碼 代碼如下:

public int hashCode() {
    int h = hash;
    if (h == 0) {

   nt off = offset;

   char val[] = value;

   int len = count;
 

   for (int i = 0; i < len; i++) {


  h = 31*h + val[off++];

   }

   hash = h;
    }
    return h;
}


解釋一下這個程序: s[0]*31^(n-1) + s[1]*31^(n-2) + ... + s[n-1] ,可以看出hash地址不一定是實際的內(nèi)存地址。

 3. 若干規(guī)范
若重寫equals(Object obj)方法,有必要重寫hashcode()方法,確保通過equals(Object obj)方法判斷結(jié)果為true的兩個對象具備相等的hashcode()返回值。說得簡單點就是:“如果兩個對象相同,那么他們的hashcode應該 相等”。不過請注意:這個只是規(guī)范,如果你非要寫一個類讓equals(Object obj)返回true而hashcode()返回兩個不相等的值,編譯和運行都是不會報錯的。不過這樣違反了Java規(guī)范,程序也就埋下了BUG。
如果equals(Object obj)返回false,即兩個對象“不相同”,并不要求對這兩個對象調(diào)用hashcode()方法得到兩個不相同的數(shù)(更印證了hash地址不一定是實際的內(nèi)存地址)。說的簡單點就是:“如果兩個對象不相同,他們的hashcode可能相同”。
根據(jù)這兩個規(guī)范,不難得到如下推論:
1、如果兩個對象equals,Java運行時環(huán)境會認為他們的hashcode一定相等。
2、如果兩個對象不equals,他們的hashcode有可能相等。
3、如果兩個對象hashcode相等,他們不一定equals(我理解是由于hash沖突造成的)。
4、如果兩個對象hashcode不相等,他們一定不equals。

三.問題解決
 測試hashCode和equals方法的使用……

復制代碼 代碼如下:

import java.util.HashMap; 
import java.util.Map;  
class A { 

    @Override 
    public boolean equals(Object obj) { 

   System.out.println("判斷equals"); 

   return true; 
    } 

    @Override 
    public int hashCode() { 

   System.out.println("判斷hashcode"); 

   return 1; 
    } 


 
public class Test { 

    public static void main(String[] args) { 

   Map<A,Object> map = new HashMap<A, Object>(); 

   map.put(new A(), new Object()); 

   map.put(new A(), new Object()); 

 


   System.out.println(map.size()); 
    } 


}


輸出:

判斷hashcode 
判斷hashcode 
判斷equals 
2


針對結(jié)果分析如下:
可以看出,JRE會調(diào)用new A()這個對象的hashcode()方法。其中:打印出的第一行“判斷hashcode”是第一次map.put(new A(), new Object())所打印出的。 接下來的“判斷hashcode”和“判斷equals”是第二次map.put(new A(), new Object())所打印出來的。當?shù)谝淮蝝ap.put(new A(), new Object())的時候,顯然,這時候沒有相同的,因為這個map中都還沒有東西,所以這時候hashcode不相等,則沒有必要再調(diào)用equals(Object obj)方法了。當?shù)诙蝝ap.put(new A(), new Object())的時候,JRE這時候發(fā)現(xiàn)了map中有兩個相同的hashcode(因為我重寫了A類的hashcode()方法永遠都返回1),所以有必要調(diào)用equals(Object obj)方法進行判斷了。然后發(fā)現(xiàn)兩個對象不equals(因為我重寫了equals(Object obj)方法,永遠都返回false)。這時候判斷結(jié)束,判斷結(jié)果:兩次存入的對象不是相同的對象。所以最后打印map的長度的時候顯示結(jié)果是:2。

四.若干注事事項
我們還應該注意,Java語言對equals()的要求如下,這些要求是必須遵循的:

對稱性:如果x.equals(y)返回是“true”,那么y.equals(x)也應該返回是“true”。
反射性:x.equals(x)必須返回是“true”。
傳遞性:如果x.equals(y)返回是“true”,而且y.equals(z)返回是“true”,那么z.equals(x)也應該返回是“true”。
一致性:如果x.equals(y)返回是“true”,只要x和y內(nèi)容一直不變,不管你重復x.equals(y)多少次,返回都是“true”。
任何情況下,x.equals(null),永遠返回是“false”;x.equals(和x不同類型的對象)永遠返回false

 以上這五點是重寫equals()方法時,必須遵守的準則,如果違反會出現(xiàn)意想不到的結(jié)果,請大家一定要遵守……

相關(guān)文章

  • 關(guān)于Spring統(tǒng)一異常處理及說明

    關(guān)于Spring統(tǒng)一異常處理及說明

    這篇文章主要介紹了關(guān)于Spring統(tǒng)一異常處理及說明,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-09-09
  • Java在重載中使用Object的問題

    Java在重載中使用Object的問題

    這篇文章主要介紹了Java在重載中使用Object的問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-02-02
  • Java中的ReentrantLock實現(xiàn)原理及代碼演示

    Java中的ReentrantLock實現(xiàn)原理及代碼演示

    這篇文章主要介紹了Java中的ReentrantLock實現(xiàn)原理及代碼演示,非公平鎖 如果已經(jīng)進入隊列,鏈表里面的線程是先進先出,如果已經(jīng)釋放了鎖,在搶占鎖時,鏈表里面的頭結(jié)點和還沒有入隊列的線程搶鎖,需要的朋友可以參考下
    2024-01-01
  • Servlet系列兩種創(chuàng)建方式

    Servlet系列兩種創(chuàng)建方式

    本文主要介紹了Servlet系列兩種創(chuàng)建方式,包含Servlet2.5之前使用和Servlet3.0后,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2024-02-02
  • Mybatis動態(tài)查詢字段及表名的實現(xiàn)

    Mybatis動態(tài)查詢字段及表名的實現(xiàn)

    本文主要介紹了Mybatis動態(tài)查詢字段及表名的實現(xiàn),通過靈活運用Mybatis提供的動態(tài)SQL功能,我們可以構(gòu)建更加靈活、高效的查詢語句,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2024-01-01
  • java創(chuàng)建excel示例(jxl使用方法)

    java創(chuàng)建excel示例(jxl使用方法)

    Java Excel是一開放源碼項目,通過它Java開發(fā)人員可以讀取Excel文件的內(nèi)容、創(chuàng)建新的Excel文件、更新 已經(jīng)存在的Excel文件。下面是使用方法,包括去掉網(wǎng)格線、字體設置、單元格設置、對齊方式等設置
    2014-03-03
  • java解決單緩沖生產(chǎn)者消費者問題示例

    java解決單緩沖生產(chǎn)者消費者問題示例

    這篇文章主要介紹了java解單緩沖生產(chǎn)者消費者問題示例,需要的朋友可以參考下
    2014-04-04
  • JVM知識總結(jié)之垃圾收集算法

    JVM知識總結(jié)之垃圾收集算法

    本博客為讀書筆記,讀的是《深入理解Java虛擬機》一書,在看這個書的時候,最大的一個感受便是“當初怎么就沒有好好學習操作系統(tǒng)呢,不然也不會有這么多看的云里霧里的地方了”,不過那都是過去的事了,學習最好的時刻便是現(xiàn)在,需要的朋友可以參考下
    2021-06-06
  • Java 17 隨機數(shù)生成器來了一波穩(wěn)穩(wěn)的增強

    Java 17 隨機數(shù)生成器來了一波穩(wěn)穩(wěn)的增強

    JDK 當中的隨機數(shù)生成器其實對于普通開發(fā)者來講基本夠用,不過對于一些比較復雜的場景來講,原有的類結(jié)構(gòu)對擴展并不是很友好,除了 Random 類,JDK 當中還提供了另外幾個隨機數(shù)的成員,下面文章將詳細介紹,需要的朋友可以參考一下
    2021-09-09
  • 解決cmd執(zhí)行javac報錯:不是內(nèi)部或外部命令,也不是可運行的程序

    解決cmd執(zhí)行javac報錯:不是內(nèi)部或外部命令,也不是可運行的程序

    剛接觸JAVA的新手可能就不知道怎么解決'JAVAC'不是內(nèi)部命令或外部命令,這篇文章主要給大家介紹了關(guān)于解決cmd執(zhí)行javac報錯:不是內(nèi)部或外部命令,也不是可運行的程序的相關(guān)資料,需要的朋友可以參考下
    2023-11-11

最新評論