真实的国产乱ⅩXXX66竹夫人,五月香六月婷婷激情综合,亚洲日本VA一区二区三区,亚洲精品一区二区三区麻豆

成都創(chuàng)新互聯(lián)網(wǎng)站制作重慶分公司

Java中如何使用HashCode

這篇文章將為大家詳細(xì)講解有關(guān)Java中如何使用HashCode,小編覺(jué)得挺實(shí)用的,因此分享給大家做個(gè)參考,希望大家閱讀完這篇文章后可以有所收獲。

成都網(wǎng)站建設(shè)哪家好,找創(chuàng)新互聯(lián)!專注于網(wǎng)頁(yè)設(shè)計(jì)、重慶網(wǎng)站建設(shè)公司、微信開(kāi)發(fā)、小程序開(kāi)發(fā)、集團(tuán)成都企業(yè)網(wǎng)站定制等服務(wù)項(xiàng)目。核心團(tuán)隊(duì)均擁有互聯(lián)網(wǎng)行業(yè)多年經(jīng)驗(yàn),服務(wù)眾多知名企業(yè)客戶;涵蓋的客戶類型包括:成都高空作業(yè)車租賃等眾多領(lǐng)域,積累了大量豐富的經(jīng)驗(yàn),同時(shí)也獲得了客戶的一致贊揚(yáng)!

關(guān)于hashCode,維基百科中:

In the Java programming language, every class implicitly or explicitly 
provides a hashCode() method, which digests the data stored in an 
instance of the class into a single hash value (a 32-bit signed 
integer).

hashCode就是根據(jù)存儲(chǔ)在一個(gè)對(duì)象實(shí)例中的所有數(shù)據(jù),提取出一個(gè)32位的整數(shù),該整數(shù)的目的是用來(lái)標(biāo)示該實(shí)例的唯一性。有點(diǎn)類似于MD5碼,每個(gè)文件都能通過(guò)MD5算法生成一個(gè)唯一的MD5碼。不過(guò),Java中的hashCode并沒(méi)有真正的實(shí)現(xiàn)為每個(gè)對(duì)象生成一個(gè)唯一的hashCode,還是會(huì)有一定的重復(fù)幾率。

先來(lái)看看Object類,我們知道,Object類是java程序中所有類的直接或間接父類,處于類層次的最高點(diǎn)。在Object類里定義了很多我們常見(jiàn)的方法,包括我們要講的hashCode方法,如下

public final native Class getClass(); 
public native int hashCode(); 
public boolean equals(Object obj) { 
 return (this == obj); 
}  
public String toString() { 
 return getClass().getName() + "@" + Integer.toHexString(hashCode()); 
}

注意到hashCode方法前面有個(gè)native的修飾符,這表示hashCode方法是由非java語(yǔ)言實(shí)現(xiàn)的,具體的方法實(shí)現(xiàn)在外部,返回內(nèi)存對(duì)象的地址。

在java的很多類中都會(huì)重寫(xiě)equals和hashCode方法,這是為什么呢?最常見(jiàn)的String類,比如我定義兩個(gè)字符相同的字符串,那么對(duì)它們進(jìn)行比較時(shí),我想要的結(jié)果應(yīng)該是相等的,如果你不重寫(xiě)equals和hashCode方法,他們肯定是不會(huì)相等的,因?yàn)閮蓚€(gè)對(duì)象的內(nèi)存地址不一樣。

public int hashCode() { 
  int h = hash; 
  if (h == 0) { 
    int off = offset; 
    char val[] = value; 
    int len = count; 

      for (int i = 0; i < len; i++) { 
        h = 31*h + val[off++]; 
      } 
      hash = h; 
    } 
    return h; 
  }

其實(shí)這段代碼是這個(gè)數(shù)學(xué)表達(dá)式的實(shí)現(xiàn)

s[0]*31^(n-1) + s[1]*31^(n-2) + … + s[n-1]

s[i]是string的第i個(gè)字符,n是String的長(zhǎng)度。那為什么這里用31,而不是其它數(shù)呢?《Effective Java》是這樣說(shuō)的:之所以選擇31,是因?yàn)樗莻€(gè)奇素?cái)?shù),如果乘數(shù)是偶數(shù),并且乘法溢出的話,信息就會(huì)丟失,因?yàn)榕c2相乘等價(jià)于移位運(yùn)算。使用素?cái)?shù)的好處并不是很明顯,但是習(xí)慣上都使用素?cái)?shù)來(lái)計(jì)算散列結(jié)果。31有個(gè)很好的特性,就是用移位和減法來(lái)代替乘法,可以得到更好的性能:31*i==(i<<5)-i?,F(xiàn)在的VM可以自動(dòng)完成這種優(yōu)化。

可以看到,String類是用它的value值作為參數(shù)來(lái)計(jì)算hashCode的,也就是說(shuō),相同的value就一定會(huì)有相同的hashCode值。這點(diǎn)也很容易理解,因?yàn)関alue值相同,那么用equals比較也是相等的,equals方法比較相等,則hashCode一定相等。反過(guò)來(lái)不一定成立。它不保證相同的hashCode一定有相同的對(duì)象。

一個(gè)好的hash函數(shù)應(yīng)該是這樣的:為不相同的對(duì)象產(chǎn)生不相等的hashCode。

在理想情況下,hash函數(shù)應(yīng)該把集合中不相等的實(shí)例均勻分布到所有可能的hashCode上,要想達(dá)到這種理想情形是非常困難的,至少java沒(méi)有達(dá)到。因?yàn)槲覀兛梢钥吹?,hashCode是非隨機(jī)生成的,它有一定的規(guī)律,就是上面的數(shù)學(xué)等式,我們可以構(gòu)造一些具有相同hashCode但value值不一樣的,比如說(shuō):Aa和BB的hashCode是一樣的。

如下代碼:

public class Main {
  public static void main(String[] args) {
    Main m = new Main();
    System.out.println(m);
    System.out.println(Integer.toHexString(m.hashCode()));
    String a = "Aa";
    String b = "BB";
    System.out.println(a.hashCode());
    System.out.println(b.hashCode());
  }
}

輸出結(jié)果:

Main@2a139a55 
2a139a55 
2112 
2112

一般在重寫(xiě)equal函數(shù)時(shí),也要重寫(xiě)hashCode函數(shù),這是為什么呢?

來(lái)看看這個(gè)例子,讓我們創(chuàng)建一個(gè)簡(jiǎn)單的類Employee

public class Employee
{
  private Integer id;
  private String firstname;
  private String lastName;
  private String department;

  public Integer getId() {
    return id;
  }
  public void setId(Integer id) {
    this.id = id;
  }
  public String getFirstname() {
    return firstname;
  }
  public void setFirstname(String firstname) {
    this.firstname = firstname;
  }
  public String getLastName() {
    return lastName;
  }
  public void setLastName(String lastName) {
    this.lastName = lastName;
  }
  public String getDepartment() {
    return department;
  }
  public void setDepartment(String department) {
    this.department = department;
  }
}

上面的Employee類只是有一些非?;A(chǔ)的屬性和getter、setter.現(xiàn)在來(lái)考慮一個(gè)你需要比較兩個(gè)employee的情形。

public class EqualsTest {
  public static void main(String[] args) {
    Employee e1 = new Employee();
    Employee e2 = new Employee();

    e1.setId(100);
    e2.setId(100);
    //Prints false in console
    System.out.println(e1.equals(e2));
  }
}

毫無(wú)疑問(wèn),上面的程序?qū)⑤敵鰂alse,但是,事實(shí)上上面兩個(gè)對(duì)象代表的是通過(guò)一個(gè)employee。真正的商業(yè)邏輯希望我們返回true。

為了達(dá)到這個(gè)目的,我們需要重寫(xiě)equals方法。

public boolean equals(Object o) {
    if(o == null)
    {
      return false;
    }
    if (o == this)
    {
      return true;
    }
    if (getClass() != o.getClass())
    {
      return false;
    }
    Employee e = (Employee) o;
    return (this.getId() == e.getId());
}

在上面的類中添加這個(gè)方法,EauqlsTest將會(huì)輸出true。

So are we done?沒(méi)有,讓我們換一種測(cè)試方法來(lái)看看。

import java.util.HashSet;
import java.util.Set;
public class EqualsTest
{
	public static void main(String[] args)
	  {
		Employee e1 = new Employee();
		Employee e2 = new Employee();
		e1.setId(100);
		e2.setId(100);
		//Prints 'true'
		System.out.println(e1.equals(e2));
		Set employees = new HashSet();
		employees.add(e1);
		employees.add(e2);
		//Prints two objects
		System.out.println(employees);
	}

上面的程序輸出的結(jié)果是兩個(gè)。如果兩個(gè)employee對(duì)象equals返回true,Set中應(yīng)該只存儲(chǔ)一個(gè)對(duì)象才對(duì),問(wèn)題在哪里呢?

我們忘掉了第二個(gè)重要的方法hashCode()。就像JDK的Javadoc中所說(shuō)的一樣,如果重寫(xiě)equals()方法必須要重寫(xiě)hashCode()方法。我們加上下面這個(gè)方法,程序?qū)?zhí)行正確。

@Override
 public int hashCode()
 {
  final int PRIME = 31;
  int result = 1;
  result = PRIME * result + getId();
  return result;
 }

需要注意記住的事情

盡量保證使用對(duì)象的同一個(gè)屬性來(lái)生成hashCode()和equals()兩個(gè)方法。在我們的案例中,我們使用員工id。
eqauls方法必須保證一致(如果對(duì)象沒(méi)有被修改,equals應(yīng)該返回相同的值)
任何時(shí)候只要a.equals(b),那么a.hashCode()必須和b.hashCode()相等。
兩者必須同時(shí)重寫(xiě)。

關(guān)于“Java中如何使用HashCode”這篇文章就分享到這里了,希望以上內(nèi)容可以對(duì)大家有一定的幫助,使各位可以學(xué)到更多知識(shí),如果覺(jué)得文章不錯(cuò),請(qǐng)把它分享出去讓更多的人看到。


文章題目:Java中如何使用HashCode
網(wǎng)站URL:http://weahome.cn/article/pijdee.html

其他資訊

在線咨詢

微信咨詢

電話咨詢

028-86922220(工作日)

18980820575(7×24)

提交需求

返回頂部