HashMap怎样解决散列(hash)冲突?

本文介绍了哈希表处理冲突的两种常见方法:链表法和开放寻址法。链表法通过每个槽位对应链表解决冲突,插入时间复杂度为O(1),查找和删除时间为O(k)。开放寻址法则是在冲突时寻找下一个空槽位,线性探测法是其一种实现,但随着冲突增加,查找效率可能降低。此外,还提到了二次探测法和双重散列法作为改进策略。
摘要由CSDN通过智能技术生成

相关视频:https://www.bilibili.com/video/BV1PQ4y1o7L4

常用两种方法:链表法开放寻址法

1、链表法(chaining)

在哈希表中,每一个桶(bucket)或者槽(slot)都会对应一条链表,所有哈希值相同的元素放到相同槽位对应的链表中。

 

在插入的时候,我们可以通过散列函数计算出对应的散列槽位,将元素插入到对应的链表即可,时间复杂度为O(1);在查找或删除元素时,我们同样通过散列函数计算出对应的散列槽位,然后再通过遍历链表进行查找或删除,时间复杂度为O(k),k为链表长度。

2、开放寻址法

核心思想:如果出现散列冲突,我们就重新探测一个空闲位置,再将元素插入。

一种比较简单的探测方法:线性探测法(Linear Probing)

但我们往散列表中插入元素时,如果某个数据经过散列函数散列之后,存储位置已经被占用了,那么我们就从当前位置开始,依次往后遍历,直到找到空余的位置插入为止(插入第一个空余的位置,方便查找),举例如图:

黄色的色块表示空余色块,橙色的色块表示已经存储了数据

在查找元素时,先将要查找元素键值通过散列函数变成散列值,然后与下标为散列值的元素比较,若相等,则说明这是我们要找的元素;若不相等,则顺序往后遍历查找,如果遍历到数组中的空余位置还是没有找到,说明要查找的元素不在散列表中。

删除元素时,删除操作不能简单地把元素设置为空,而是要特殊标记为deleted,因为如果简单设置为空,在查找元素的过程中遇到这个被删除元素的位置就会停下,而不是继续往后遍历,会使查找算法失效;但是如果特色标记为deleted,当线性探测查找时,遇到标记为deleted的位置就会往下探测。

线性探测法的缺点:当插入的数据越来越多时,散列冲突发生的可能性会越来越大,空余位置会越来越少,线性探测的时间会越来越长,最坏时间复杂度为O(n)。

另外的两种探测方法是二次探测法(Quadratic probing)和双重散列法(Double hashing)。

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值