Java三十二篇: 哈希表


Java三十二篇: 哈希表

平安春运


1.概念

哈希表(Hash table,也叫散列表):

是根据关键码值(Key value)而直接进行访问的数据结构。也就是说,它通过把关键码值映射到表中一个位置来访问记录,以加快查找的速度。这个映射函数叫做散列函数,存放记录的数组叫做散列表。

给定表M,存在函数f(key),对任意给定的关键字值key,代入函数后若能得到包含该关键字的记录在表中的地址,则称表M为哈希(Hash)表,函数f(key)为哈希(Hash) 函数。

哈希表的本质上上一个数组(元素是Entry)

这里的 id 是个key,哈希表就是根据key值来通过哈希函数计算得到一个值,这个值就是用来确定这个Entry要存放在哈希表中的位置的,实际上这个值就是一个下标值,来确定放在数组的哪个位置上

比如这里的 id 是001,那么经过哈希函数的计算之后得到了1,这个1就是告诉我们应该把这个Entry放到哪个位置,这个1就是数组的确切位置的下标,也就是需要放在数组中下表为1的位置

Java三十二篇: 哈希表
img

Hash Table的查询速度非常的快,几乎是O(1)的时间复杂度

hash就是找到一种数据内容和数据存放地址之间的映射关系

散列法:元素特征转变为数组下标的方法

优点:

不论哈希表中有多少数据,查找、插入、删除(有时包括删除)时间接近常量的时间即0(1)的时间级

缺点:

它是基于数组的,数组创建后难于扩展,某些哈希表被基本填满时,性能下降得非常严重,所以程序员必须要清楚表中将要存储多少数据

2.哈希函数的构造方法

2.1 直接定制法(不常用)

取关键字或关键字的某个线性函数值为哈希地址 即, H(key) = key H(key) = a * key + b

优点:简单,均匀,不会产生冲突 缺点:需要实现直到关键字的分布情况,适合查找表比较小且连续的情况

2.2 数字分析法

数字分析法用于处理关键字是位数比较多的数字,通过抽取关键字的一部分进行操作,计算哈希存储位置的方法

例:

身份证号是有规律的,现在要存储一个班级学生的身份证号码,假设这个班级的学生都出生在同一个地区,同一年,那么他们的身份证的前面数位都是相同的,那么我们可以截取后面不同的几位存储,假设有5位不同,那么就用这五位代表地址

适用场景:

处理关键字位数比较大的情况,事先知道关键字的分布且关键字的若干位分布均匀


2.3 平方取中法

先对关键字取平方,然后选取中间几位为哈希地址,取的位数由表长决定

例:key=1234 1234^2=1522756 取227作hash地址 key=4321 4321^2=18671041 取671作hash地址

适用场景:不知道关键字的分布,而位数又不是很大的情况

2.4 折叠法

如果数字的位数很多,可以将数字分割为几个部分,取他们的叠加和作为hash地址

例:key=123 456 789 可以存储在61524,取末三位,存在524的位置

适用场景:关键字位数很多,而且关键字每一位上数字分布大致均匀

2.5 除留余数法(用的较多)

H(key)=key MOD p (p<=m m为表长)

例:存储3 6 9,那么p就不能取3 因为 3 MOD 3 == 6 MOD 3 == 9 MOD 3,地址冲突

一般来说,p应为不大于m的质数或是不含20以下的质因子的合数,这样可以减少地址的重复(冲突)

2.6 随机数法

选择一个随机数,取关键字的随机函数值作为他的哈希地址 即,f(key) = random (key)

适合场景:关键字的长度不等时。当遇到特殊字符的关键字时,需要将其转换为某种数字

3.哈希冲突及解决方式

哈希冲突图例:
Java三十二篇: 哈希表
img

3.1 开放寻址法

H(key)的哈希函数:

H(key1)= H(keyi)

那么 keyi 存储位置 Hi = (H(key)+di) MOD m ,m为表长

di 有三种取法:

线性探测再散列: di = c * i

平方探测再散列:di = 1^2 , -1 ^2,2 ^2,-2 ^2…

随机探测再散列(双探测再散列):di是一组伪随机数列

简单来说就是,既然位置被占了,那就另外再找个位置,怎么找其他的位置呢?这里其实也有很多的实现,我们说个最基本的就是既然当前位置被占用了,我们就看看该位置的后一个位置是否可用,也就是1的位置被占用了,我们就看看2的位置,如果没有被占用,那就放到这里呗,当然,也有可能2的位置也被占用了,那咱就继续往下找,看看3的位置,一次类推,直到找到空位置

3.2 链地址法

存储数据后面加一个指针,指向后面冲突的数据

比如:

Java三十二篇: 哈希表
img

简单来说,Entry还额外的保存了一个next指针,这个指针指向数组外的另外一个位置,将李四安排在这里,然后张三那个Entry中的next指针就指向李四的这个位置,也就是保存的这个位置的内存地址,如果还有冲突,那就把又冲突的那个Entry放在一个新位置上,然后李四的Entry中的next指向它,这样就形成了一个链表。


Java三十二篇: 哈希表
img

3.3 再哈希法

Hi = RHi(key) i = 1,2,…,k RHi均是不同的哈希函数,意思为:当繁盛冲突时,使用不同的哈希函数计算地址,直到不冲突为止。这种方法不易产生堆积,但是耗费时间

3.4 公共溢出区法

即设立两个表:基础表和溢出表。将所有关键字通过哈希函数计算出相应的地址。然后将未发生冲突的关键字放入相应的基础表中,一旦发生冲突,就将其依次放入溢出表中即可。

在查找时,先用给定值通过哈希函数计算出相应的散列地址后,首先与基本表的相应位置进行比较,如果不相等,再到溢出表中顺序查找。

此种方法适用于数据和冲突较少的情况



4.扩容

增长因子,也叫作负载因子,简单点说就是已经被占的位置与总位置的一个百分比。

比如负载因子是0.7时,一共十个位置,现在已经占了七个位置,就触发了扩容机制,也就是达到了总位置的百分之七十就需要扩容。

简单来说,扩容就是新创建一个数组是原来的2倍,然后把原数组的所有Entry都重新Hash一遍放到新的数组。

数组扩大了,所以一般哈希函数也会有变化,这里的Hash也就是把之前的数据通过新的哈希函数计算出新的位置来存放



代码的实际需求

  • 看一个实际需求, google 公司的一个上机题:

  • 有一个公司,当有新的员工来报道时,要求将该员工的信息加入(id, 性别, 年龄, 住址…),当输入该员工的 id 时,要求查找到该员工的所有信息

  • 要求:不使用数据库,尽量节省内存,速度越快越好 => 哈希表(散列)

哈希表编程思路:


    • 先根据对象的信息将其散列,得到 hashCode
    • 根据对象的 hashCode 值,找到对应的数组下标,其实就是找到存储对象的链表

    • 在链表中进行相应的增删改查操作
代码实现
Emp 节点
add:  添加雇员
list: 显示雇员
find: 查找雇员
exit: 退出系统

add
输入id
1
输入名字
Heygo
add: 添加雇员
list: 显示雇员
find: 查找雇员
exit: 退出系统

list
1 链表为空
2 链表的信息为 => id=1 name=Heygo => id=8 name=NiuNiu
3 链表的信息为 => id=2 name=Oneby
4 链表为空
5 链表为空
6 链表为空
7 链表为空
add: 添加雇员
list: 显示雇员
find: 查找雇员
exit: 退出系统

find
请输入要查找的id
9
在哈希表中,没有找到该雇员~
add: 添加雇员
list: 显示雇员
find: 查找雇员
exit: 退出系统

总结:

哈希表基于数组,类似于key-value的存储形式,关键字值通过哈希函数映射为数组的下标,如果一个关键字哈希化到已占用的数组单元,这种情况称为冲突。用来解决冲突的有两种方法:开放地址法和链地址法。在开发地址法中,把冲突的数据项放在数组的其它位置;在链地址法中,每个单元都包含一个链表,把所有映射到同一数组下标的数据项都插入到这个链表中。

Java三十二篇: 哈希表
img

 


本篇文章来源于微信公众号: 小刘Java之路

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

文章由极客之音整理,本文链接:https://www.bmabk.com/index.php/post/11112.html

(0)
小半的头像小半

相关推荐

发表回复

登录后才能评论
极客之音——专业性很强的中文编程技术网站,欢迎收藏到浏览器,订阅我们!