一文告訴你為什么要重寫hashCode()方法和equals()方法
首先我們看下object源碼中如何定義hashcode與equals方法的
public native int hashCode();
public boolean equals(Object obj) {
return (this == obj);
}
Object類中的hashCode()方法,用的是native關(guān)鍵字修飾,說明這個(gè)方法是個(gè)原生函數(shù),也就說這個(gè)方法的實(shí)現(xiàn)不是用java語言實(shí)現(xiàn)的,是使用c/c++實(shí)現(xiàn)的,并且被編譯成了DLL,由java去調(diào)用,jdk源碼中不包含。
Java將調(diào)用本地方法庫(kù)對(duì)此方法的實(shí)現(xiàn)。由于Object類中有JNI方法調(diào)用,按照J(rèn)NI的規(guī)則,應(yīng)當(dāng)生成JNI 的頭文件,在此目錄下執(zhí)行 javah -jni java.lang.Object 指令,將生成一個(gè) java_lang_Object.h 頭文件
/* * Class: java_lang_Object * Method: hashCode * Signature: ()I */ JNIEXPORT jint JNICALL Java_java_lang_Object_hashCode (JNIEnv *, jobject);
具體在c中怎么實(shí)現(xiàn)我也不是很清楚
但是為什么要重寫equals與hashcode呢,看個(gè)例子
先定義一個(gè)實(shí)體類
package org;
public class Chengxuyuan {
private Integer age;
private String company;
public Integer getAge() {
return age;
}
public void setAge(Integer age) {
this.age = age;
}
public String getCompany() {
return company;
}
public void setCompany(String company) {
this.company = company;
}
@Override
public String toString() {
return "Chengxuyuan{" +
"age=" + age +
", company='" + company + '\'' +
'}';
}
// @Override
// public int hashCode() {
// int hashcode = age.hashCode();
// hashcode = 31 * hashcode + company.hashCode();
// return hashcode;
// }
//
// @Override
// public boolean equals(Object obj) {
// if(!(obj instanceof Chengxuyuan)) {//首先要判斷是不是同一個(gè)類型
// return false;
// }
// Chengxuyuan chengxuyuan = (Chengxuyuan) obj;
// if (this == chengxuyuan) {// 其次要判斷地址是否相同相等
// return true;
// }
// if (chengxuyuan.age.equals(this.age) && chengxuyuan.company.equals(this.company)) {// 最后要判斷對(duì)象里的屬性是否相同
// return true;
// } else {
// return false;
// }
// }
}
然后通過對(duì)map的操作來查看結(jié)果
package org;
import java.util.HashMap;
import java.util.Map;
public class Main {
public static void main(String[] args) {
Map<Chengxuyuan, String> map = new HashMap<>();
Chengxuyuan chengxuyuan = new Chengxuyuan();
chengxuyuan.setAge(15);
chengxuyuan.setCompany("沒公司");
System.out.println(chengxuyuan.hashCode());
map.put(chengxuyuan, chengxuyuan.getCompany());
Chengxuyuan chengxuyuan2 = new Chengxuyuan();
chengxuyuan2.setAge(15);
chengxuyuan2.setCompany("沒公司");
System.out.println(chengxuyuan2.hashCode());
map.put(chengxuyuan2, chengxuyuan2.getCompany());
System.out.println(chengxuyuan.equals(chengxuyuan2));
System.out.println(map);
System.out.println(map.get(chengxuyuan2));
}
}
查看結(jié)果
824318946
930990596
false
{Chengxuyuan{age=15, company='沒公司'}=沒公司, Chengxuyuan{age=15, company='沒公司'}=沒公司}
沒公司
從上述內(nèi)容可以看到 同樣內(nèi)容保存到map中本應(yīng)該是一條內(nèi)容,但是現(xiàn)在是兩條信息,在map中保存數(shù)據(jù),首先hashmap在保存數(shù)據(jù)的時(shí)候會(huì)會(huì)計(jì)算key的hashcode來作為key的鍵值來保存信息
hashmap源碼
public V put(K key, V value) {
return putVal(hash(key), key, value, false, true);
}
所以當(dāng)key得hashcode值沒有被重寫的話兩個(gè)對(duì)象不是是無法相等的,所以首先要重寫hashcode
重寫hashcode
/**
* 根據(jù)指定數(shù)組的內(nèi)容返回哈希碼。如果該數(shù)組包含其他數(shù)組作為元素,
* 則哈希碼基于其標(biāo)識(shí)而不是其內(nèi)容。因此,可以直接或間接通過一個(gè)
* 或多個(gè)級(jí)別的數(shù)組在包含自身作為元素的數(shù)組上調(diào)用此方法。 <p>對(duì)
* 于任何兩個(gè)數(shù)組<tt> a <tt>和<tt> b <tt>,例如<tt> Arrays.equals
* (a,b)<tt>,<tt> Arrays也是這種情況。 hashCode(a)==
* Arrays.hashCode(b)<tt>。 <p>此方法返回的值等于<tt>
* Arrays.asList(a).hashCode()<tt>返回的值,除非<tt> a <tt>為<tt>
* null < tt>,在這種情況下,返回<tt> 0 <tt>。 @param一個(gè)數(shù)組,
* 其數(shù)組基于內(nèi)容的哈希碼來計(jì)算@返回<tt> a <tt>的基于內(nèi)容的哈希碼
* @see deepHashCode(Object [])@ 1.5起
*
*/
public static int hashCode(Object a[]) {
if (a == null)
return 0;
int result = 1;
for (Object element : a)
result = 31 * result + (element == null ? 0 : element.hashCode());
return result;
}
通過上述方式重寫hashcode,但是這樣寫也會(huì)有沖突,個(gè)人認(rèn)為最好是單個(gè)元素計(jì)算hashcode,例如將每個(gè)變量疊加求md5值以保證唯一性(在不確定實(shí)體類中變量在應(yīng)用的時(shí)候唯一)
但是只重寫hashcode值這樣并不能保證map的在保存的時(shí)候能夠唯一
將上述實(shí)體類中的重寫hashcode注釋打開,發(fā)現(xiàn)hash值相同但 比較的時(shí)候并不相同
27392574
27392574
false
{Chengxuyuan{age=15, company='沒公司'}=沒公司, Chengxuyuan{age=15, company='沒公司'}=沒公司}
沒公司
所以需要重寫equals方法
27392574
27392574
false
{Chengxuyuan{age=15, company='沒公司'}=沒公司, Chengxuyuan{age=15, company='沒公司'}=沒公司}
沒公司
再次運(yùn)行兩個(gè)對(duì)象就相同了
27392574
27392574
true
{Chengxuyuan{age=15, company='沒公司'}=沒公司}
沒公司
這樣就可以插入到map中了 通過實(shí)體類的可以就可以獲取到元素
到此這篇關(guān)于一文告訴你為什么要重寫hashCode()方法和equals()方法的文章就介紹到這了,更多相關(guān)Java重寫hashCode()方法和equals()方法內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
解決CollectionUtils.isNotEmpty()不存在的問題
這篇文章主要介紹了解決CollectionUtils.isNotEmpty()不存在的問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-02-02
SpringCloud聲明式Feign客戶端調(diào)用工具使用
這篇文章主要為大家介紹了SpringCloud聲明式Feign客戶端調(diào)用工具使用詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-08-08
CommonMark 使用教程:將 Markdown 語法轉(zhuǎn)成 Html
這篇文章主要介紹了CommonMark 使用教程:將 Markdown 語法轉(zhuǎn)成 Html,這個(gè)技巧我們做任何網(wǎng)站都可以用到,而且非常好用。,需要的朋友可以參考下2019-06-06
spring cloud hystrix 超時(shí)時(shí)間使用方式詳解
這篇文章主要介紹了spring cloud hystrix 超時(shí)時(shí)間使用方式,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-01-01
spring cloud gateway跨域全局CORS配置方式
這篇文章主要介紹了spring cloud gateway跨域全局CORS配置方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-07-07
JAVA的LIST接口的REMOVE重載方法調(diào)用原理解析
這篇文章主要介紹了JAVA的LIST接口的REMOVE重載方法調(diào)用原理解析,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-10-10
java IO流將一個(gè)文件拆分為多個(gè)子文件代碼示例
這篇文章主要介紹了java IO流將一個(gè)文件拆分為多個(gè)子文件代碼示例,具有一定借鑒價(jià)值,需要的朋友可以參考下。2017-12-12
netty?pipeline中的inbound和outbound事件傳播分析
這篇文章主要為大家介紹了netty?pipeline中的inbound和outbound事件傳播分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-04-04
java的main方法中調(diào)用spring的service方式
這篇文章主要介紹了在java的main方法中調(diào)用spring的service方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-12-12

