很多人問復(fù)寫equals一定要復(fù)寫hashCode方法嗎?
我們在判斷對象是否相等的時(shí)候往往會(huì)用的equals方法,根據(jù)對象的某個(gè)值是否相同來決定對象是否相等,所以這里我們需要復(fù)寫equals方法。例如下面的代碼:
public class Persion { PRivate Integer id; private Integer age; private String name; @Override public boolean equals(Object obj) { if(this.getClass()==obj.getClass()){ return id.equals(((Persion)obj).id); } return false; } public Integer getId() { return id; } public void setId(Integer id) { this.id = id; } public Integer getAge() { return age; } public void setAge(Integer age) { this.age = age; } public String getName() { return name; } public void setName(String name) { this.name = name; }} public static void main(String[] args) { Persion p1 = new Persion(); p1.setId(1); p1.setAge(1); p1.setName("張三"); Persion p2 = new Persion(); p2.setId(1); p2.setAge(2); p2.setName("李四"); System.out.println(p1.equals(p2)); } //返回的是true那么問題來了,既然equals可以實(shí)現(xiàn)功能我們?yōu)槭裁催€要復(fù)寫hascCode,接著往下看:
@Test public void testHashCode(){ HashMap<Persion, String> map1 = new HashMap<>(); Persion p1 = new Persion(); p1.setId(1); p1.setAge(1); p1.setName("張三"); map1.put(p1, "測試"); Persion p2 = new Persion(); p2.setId(1); p2.setAge(2); p2.setName("李四"); System.out.println(map1.containsKey(p2)); } //返回的是false因?yàn)閔ashMap(或是hashSet)的containsKey方法涉及到了hashCode的操作,
public boolean containsKey(Object paramObject) { return getNode(hash(paramObject), paramObject) != null; } static final int hash(Object paramObject) { int i; return paramObject == null ? 0 : (i = paramObject.hashCode()) ^ i >>> 16; }這里如果想返回true,必須equals與hashCode同時(shí)返回true才行。
我們復(fù)寫Persion的hashCode方法
@Override public int hashCode() { return id.hashCode(); }再執(zhí)行測試用例就可以返回true.
總結(jié):理論上沒有涉及到hashCode的操作,我們復(fù)寫equals方法就可以,但是,java規(guī)范約定,如果重寫equals方法,那也要重寫hashCode方法,使equals為真的情況,hashCode的值也是相同的,所以我們在復(fù)寫的equals的時(shí)候盡量也復(fù)寫hashCode,保險(xiǎn)一點(diǎn)。
補(bǔ)充:String與Integer的hashCode是根據(jù)他們包內(nèi)的值來計(jì)算生成的,如果兩個(gè)值相等則hashCode相等。
String a = "張三"; String b = "張三"; System.out.println(a.hashCode()==b.hashCode()); Integer c =1; Integer d =1; System.out.println(c.hashCode()==d.hashCode());新聞熱點(diǎn)
疑難解答