javaee论坛

普通会员

225648

帖子

355

回复

369

积分

楼主
发表于 2019-11-07 13:03:31 | 查看: 94 | 回复: 2

HashMap的结构是

包含多个HashCode的数组,每个HashCode对应一个Bucket(桶),每个桶是一个LinkList(链表),链表里面存储的是多个Entry(键值对+next+hash值)

数组中存取值通过如下算法:

//存储时:inthash=key.hashCode();intindex=hash%Entry[].length;Entry[index]=value;//取值时:inthash=key.hashCode();intindex=hash%Entry[].length;returnEntry[index];

HashMpa.put(对象)

对象就是一个Entry,程序员控制键值对的内容,next和hash会自动生成

碰撞:当系统给两个Entry生成的hash相同时就会发生碰撞,最后Entry,会存储到相同hash对应的链表里,其中next,就是标记多个相同hash的Entry之间的链表连接关系的。

next和hash是怎么自动生成的?

put()当程序员put一个Entry时,先使用Hash函数计算出下数组下标,确定这个Entry插入到hashcode数组的哪个位置publicVput(Kkey,Vvalue){if(key==null)returnputForNullKey(value);//null总是放在数组的第一个链表中inthash=hash(key.hashCode());inti=indexFor(hash,table.length);//遍历链表for(Entry<K,V>e=table[i];e!=null;e=e.next){Objectk;//如果key在链表中已存在,则替换为新valueif(e.hash==hash&&((k=e.key)==key||key.equals(k))){VoldValue=e.value;e.value=value;e.recordAccess(this);returnoldValue;}}modCount++;addEntry(hash,key,value,i);returnnull;}voidaddEntry(inthash,Kkey,Vvalue,intbucketIndex){Entry<K,V>e=table[bucketIndex];table[bucketIndex]=newEntry<K,V>(hash,key,value,e);//参数e,是Entry.next//如果size超过threshold,则扩充table大小。再散列if(size++>=threshold)resize(2*table.length);}get()publicVget(Objectkey){if(key==null)returngetForNullKey();inthash=hash(key.hashCode());//先定位到数组元素,再遍历该元素处的链表for(Entry<K,V>e=table[indexFor(hash,table.length)];e!=null;e=e.next){Objectk;if(e.hash==hash&&((k=e.key)==key||key.equals(k)))returne.value;}returnnull;}

根据Key值做一次Hash映射,算出其对应的数组下标值,碰撞问题,同一个位置的链表里可能存在多个Entry,这时就要从对应链表的头节点开始,一个个向下查找,直到找到对应的Key值,这样就获得到了所要查找的键值对

hashcode数组的长度?有一个初始长度16,每当触发到75%,长度扩大一倍,然后把之前的数组全部数据移动到新数组中。

hashmap死锁在多线程的情况下,当重新调整HashMap大小的时候,就会存在条件竞争,因为如果两个线程都发现HashMap需要重新调整大小了,它们会同时试着调整大小

使用hashtable(完全同步)和currentHashMap(分片同步)等线程安全的可以避免死锁。


普通会员

1

帖子

355

回复

376

积分
沙发
发表于 2022-04-20 14:58:59

看看

普通会员

0

帖子

319

回复

324

积分
板凳
发表于 2024-05-08 06:42:35

如果你智商能再高点,也许我会上当

您需要登录后才可以回帖 登录 | 立即注册

触屏版| 电脑版

技术支持 历史网 V2.0 © 2016-2017