亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

JAVA hashCode使用方法詳解

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

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

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

復(fù)制代碼 代碼如下:

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

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

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

復(fù)制代碼 代碼如下:

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;
}


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

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

三.問(wèn)題解決
 測(cè)試hashCode和equals方法的使用……

復(fù)制代碼 代碼如下:

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


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

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

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

 以上這五點(diǎn)是重寫(xiě)equals()方法時(shí),必須遵守的準(zhǔn)則,如果違反會(huì)出現(xiàn)意想不到的結(jié)果,請(qǐng)大家一定要遵守……

相關(guān)文章

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

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

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

    Java在重載中使用Object的問(wèn)題

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

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

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

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

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

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

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

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

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

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

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

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

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

    Java 17 隨機(jī)數(shù)生成器來(lái)了一波穩(wěn)穩(wěn)的增強(qiáng)

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

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

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

最新評(píng)論