HashMap實(shí)現(xiàn)保存兩個(gè)key相同的數(shù)據(jù)
HashMap如何保存兩個(gè)key相同的數(shù)據(jù)
最近一個(gè)朋友去面試了,面試官問(wèn)了一個(gè)關(guān)于HashMap的問(wèn)題:HashMap如何保存兩個(gè)key相同的數(shù)據(jù)?
準(zhǔn)確來(lái)說(shuō),應(yīng)該是Map中如何保存兩個(gè)key相同的數(shù)據(jù),因?yàn)橛脕?lái)實(shí)現(xiàn)這個(gè)功能的IdentityHashMap類(lèi)和HashMap雖然都是實(shí)現(xiàn)了Map接口,但本質(zhì)是屬于不同的東西;
我們知道在HashMap中,如果key相同就會(huì)被覆蓋,那IdentityHashMap是怎么實(shí)現(xiàn)這個(gè)功能的呢?
java jdk源碼中,IdentityHashMap類(lèi)上寫(xiě)了100來(lái)行注釋的代碼,如果用一句話來(lái)總結(jié)的話:
IdentityhashMap類(lèi)利用哈希表實(shí)現(xiàn)Map接口,比較鍵(和值)時(shí)使用引用相等性代替對(duì)象相等性,也就是說(shuō)key(value)比較的時(shí)候只比較兩個(gè)key是否引用同一個(gè)對(duì)象,比較的是對(duì)象的地址;
測(cè)試1:
public static void main(String[] args) { String str1 = "key"; String str2 = "key"; System.out.println(str1==str2); Map<String, String> map = new IdentityHashMap<>(); map.put(str1, "value1"); map.put(str2, "value2"); map.forEach((k,v)-> System.out.println(k+"->"+v)); }
打印:
true
key->value2
測(cè)試1中,將字符串"key"直接賦值給str1和str2,因?yàn)樽址欠旁诔A砍刂械?,所以str1和str2實(shí)際上還是同一個(gè)對(duì)象,所以它們的key值是相同的,會(huì)被覆蓋;
測(cè)試2:
public static void main(String[] args) { String str1 = new String("key"); String str2 = new String("key"); System.out.println(str1==str2); Map<String, String> map = new IdentityHashMap<>(); map.put(str1, "value1"); map.put(str2, "value2"); map.forEach((k,v)-> System.out.println(k+"->"+v)); }
打印:
false
key value1
key value2
測(cè)試2中,str1和str2是通過(guò)new的方式創(chuàng)建出來(lái)的,屬于不同對(duì)象,所以它們的引用不同,key值也就不同,所以put的時(shí)候不會(huì)被覆蓋;
關(guān)于IdentityHashMap常不常用,實(shí)際開(kāi)發(fā)中我基本沒(méi)用過(guò),所以在什么場(chǎng)景會(huì)用到IdentityHashMap我也說(shuō)不出個(gè)一二來(lái);不過(guò)存在即合理,肯定有什么場(chǎng)景會(huì)用到的,后面遇到我會(huì)及時(shí)更新~
HashMap插入相同key
使用HashMap在插入操作時(shí),會(huì)通過(guò)equal方法判斷key是否相同。如果相同,則將覆蓋對(duì)應(yīng)的value;不相同才使用新的“桶”。
我的問(wèn)題
當(dāng)往HashMap中插入數(shù)據(jù),即使有相同的key,但是能不能不進(jìn)行覆蓋操作,而是把新的value放在原有的value附近能夠找到的位置?
想法
呃,其實(shí)大概方向就是通過(guò)一個(gè)HashMap<Integer, ArrayList>實(shí)現(xiàn)。。。
貼上代碼
import java.util.ArrayList; import java.util.HashMap; public class MapAndLink { public static void main(String[] args){ HashMap<Integer, ArrayList> map = new HashMap<>(); put(1, 1, map); put(1, 3, map); put(2, 2, map); put(3, 4, map); put(1, 3, map); System.out.println(map.toString()); } public static void put(Integer key, Integer str, HashMap<Integer, ArrayList> map){ ArrayList<Integer> list = map.get(key); if(list == null) list = new ArrayList(); for(int i = 0; i < list.size(); ++i){ if(list.get(i).equals(str)) return; } list.add(str); map.put(key, list); } }
再貼上輸出結(jié)果
{1=[1, 3], 2=[2], 3=[4]}
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
Java基于循環(huán)遞歸回溯實(shí)現(xiàn)八皇后問(wèn)題算法示例
這篇文章主要介紹了Java基于循環(huán)遞歸回溯實(shí)現(xiàn)八皇后問(wèn)題算法,結(jié)合具體實(shí)例形式分析了java的遍歷、遞歸、回溯等算法實(shí)現(xiàn)八皇后問(wèn)題的具體步驟與相關(guān)操作技巧,需要的朋友可以參考下2017-06-06Mybatisplus創(chuàng)建Spring?Boot工程打包錯(cuò)誤的解決方式
最近在實(shí)戰(zhàn)springboot遇到了一些坑,記錄一下,下面這篇文章主要給大家介紹了關(guān)于Mybatisplus創(chuàng)建Spring?Boot工程打包錯(cuò)誤的解決方式,文中通過(guò)圖文介紹的介紹的非常詳細(xì),需要的朋友可以參考下2023-03-03出現(xiàn)java.lang.NoSuchMethodException異常的解決(靠譜)
這篇文章主要介紹了出現(xiàn)java.lang.NoSuchMethodException異常的解決方案(靠譜),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-03-03SpringBoot java-jar命令行啟動(dòng)原理解析
這篇文章主要介紹了SpringBoot java-jar命令行啟動(dòng)原理解析,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-07-07java中排序報(bào):Comparison method violates its general contract異常的解
這篇文章主要給大家介紹了關(guān)于java中排序報(bào):Comparison method violates its general contract異常的解決方法,文中介紹的非常詳細(xì),對(duì)大家具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來(lái)一起看看吧。2017-06-06