【数据结构】关于哈希表内部原理,你到底了解多少???(超详解)

2024-08-31 06:44

本文主要是介绍【数据结构】关于哈希表内部原理,你到底了解多少???(超详解),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

前言:

🌟🌟本期讲解关于哈希表的内部实现原理,希望能帮到屏幕前的你。

🌈上期博客在这里:http://t.csdnimg.cn/7D225

🌈感兴趣的小伙伴看一看小编主页:GGBondlctrl-CSDN博客

目录

 📚️1.哈希表的概念

📚️2.哈希-冲突

2.1冲突-概念

 2.2冲突-避免

1.冲突-避免-哈希函数设计

2.冲突-避免-冲突因子

2.3冲突-解决

1.冲突-解决-闭散列

2.冲突-解决-开散列

 2.4性能分析

2.5与Java类集的关系

 📚️3.总结


 📚️1.哈希表的概念

       顺序结构以及平衡树中在查找一个元素时,必须要经过关键码的多次比较。顺序查找时间复杂度为O(N);

      平衡树中为树的高度,即O(logN ),搜索的效率取决于搜索过程中元素的比较次数。例如上期的treeMap.

理想的搜索方法:可以不经过任何比较,一次直接从表中得到要搜索的元素。 如果构造一种存储结构,通过某种函数(hashFunc)使元素的存储位置与它的关键码之间能够建立一一映射的关系,那么在查找时通过该函数可以很快找到该元素。

  • 插入元素
根据待插入元素的关键码,以此函数计算出该元素的存储位置并按此位置进行存放
  • 搜索元素
对元素的关键码进行同样的计算,把求得的函数值当做元素的存储位置,在结构中按此位置取元素比较,若关键码相等,则搜索成功


方式即为哈希(散列)方法,哈希方法中使用的转换函数称为哈希(散列)函数,构造出来的结构称为哈希表(HashTable)(或者称散列表)

 哈希函数设置为:hash(key) = key % capacity; capacity为存储元素底层空间总的大小。

图解 :

注意:这里和计数排序差不多 ,但是如果我们加入11,会发现11的位置将直接把1给覆盖掉,那么此时就叫作哈希-冲突。

📚️2.哈希-冲突

2.1冲突-概念

对于两个数据元素的关键字ki 和kj (i != j),有ki !=kj ,但有:Hash(ki ) == Hash(kj ),即:不同关键字通过相同哈希哈数计算出相同的哈希地址,该种现象称为哈希冲突或哈希碰撞

 2.2冲突-避免

首先,我们需要明确一点,由于我们哈希表底层数组的容量往往是小于实际要存储的关键字的数量的,这就导致一个问题,冲突的发生是必然的,但我们能做的应该是尽量的降低冲突率

1.冲突-避免-哈希函数设计

哈希函数设计原理:

• 哈希函数的定义域必须包括需要存储的全部关键码,而如果散列表允许有m个地址时,其值域必须在0到m-1之间


• 哈希函数计算出来的地址能均匀分布在整个空间中


• 哈希函数应该比较简单

小编这里介绍两个比较常用的两个方法: 

1. 直接定制法
       取关键字的某个线性函数为散列地址:Hash(Key)= A*Key + B 优点:简单、均匀 缺点:需要事先知道关键字的分布情况 使用场景:适合查找比较小且连续的情况。

例如字符串中第一个只出现一次字符

代码实例如下:

class Solution {public int firstUniqChar(String s) {int[] array=new int[26];for(int i=0;i<s.length();i++){char ch=s.charAt(i);array[ch-'a']++;}for(int j=0;j<s.length();j++){if(array[s.charAt(j)-'a']==1){return j;}}return -1;}
}

思路:就是创建一个26个空间大小的数组,在遍历字符串时,将字符对应的位置加一,最后再次遍历谁为1,就返回第一个不重复的字符。 

注意:这种方法只适合要求范围小的数值内进行操作,如果范围过大,则会造成数组空间的浪费。

 2. 除留余数法
      设散列表中允许的地址数为m,取一个不大于m,但最接近或者等于m的质数p作为除数,按照哈希函数:Hash(key) = key% p(p<=m),将关键码转换成哈希地址

2.冲突-避免-冲突因子

负载因子与冲突率图片演示:

 

 已知哈希表中已有的关键字个数是不可变的,那我们能调整的就只有哈希表中的数组的大小。(我们不可能在源头上限制我们的需求)

2.3冲突-解决

1.冲突-解决-闭散列

闭散列:也叫开放定址法,当发生哈希冲突时,如果哈希表未被装满,说明在哈希表中必然还有空位置,那么可以把key存放到冲突位置中的“下一个” 空位置中去。

线性探测:

这里就是,当我们先插入了一个1后,然后想插入11,此时发现取余地址冲突了,此时我们就要往后寻空位置,发现后进行插入。

图片演示:

注意:

      不能随便物理删除哈希表中已有的元素,若直接删除元素会影响其他元素的搜索。比如删除元素4,如果直接删除掉,44查找起来可能会受影响。

      产生冲突的数据堆积在一块

二次探测

找下一个空位置的方法为:Hi = (H0 +i^2 )% m, 或者:Hi= (H0 -i^2 )% m。其中:i = 1,2,3…, 是通过散列函数Hash(x)对元素的关键码 key 进行计算得到的置,m是表的大小。

图片演示:

当然这是一个极端的例子~~~ 

注意:比散列最大的缺陷就是空间利用率比较低,这也是哈希的缺陷。

2.冲突-解决-开散列

开散列法又叫链地址法(开链法),首先对关键码集合用散列函数计算散列地址,具有相同地址的关键码归于同一子集合,每一个子集合称为一个桶,各个桶中的元素通过一个单链表链接起来,各链表的头结点存储在哈希表中。

图解如下:

 注意:刚才我们提到了,哈希桶其实可以看作将大集合的搜索问题转化为小集合的搜索问题了,那如果冲突严重,就意味着小集合的搜索性能其实也时不佳的,这个时候我们就可以将这个所谓的小集合搜索问题继续进行转化,例如:
1. 每个桶的背后是另一个哈希表
2. 每个桶的背后是一棵搜索树

 开散列的代码实现模拟:

public class HashBucket {//结点链表的初始化static class Node{public int key;public int val;public Node next;public Node(int key,int val){this.key=key;this.val=val;}}//数组的初始化public Node[] array=new Node[10];public int usedSize;//插入元素public void put(int key,int value){/* int index=key% array.length;//遍历链表//头插法Node cur=array[index];while (cur!=null){if(cur.key==key){cur.val=value;return;}cur=cur.next;}Node node=new Node(key,value);node.next=array[index];array[index]=node;usedSize++;*///尾插Node node=new Node(key,value);int index=key% array.length;if(array[index]==null){array[index]=node;return;}Node cur=array[index];Node prev=null;while (cur!=null){if(cur.key==key){cur.val=value;return;}prev=cur;cur=cur.next;}prev.next=node;usedSize++;//负载因子是否大于0.75if(loadFactor()>0.75){resize();}}public int loadFactor(){return usedSize/ array.length;}//进行扩容public void resize(){Node[] tmpArray=new Node[array.length*2];for (int i = 0; i < array.length; i++) {Node cur=array[i];while (cur!=null){Node newCur=cur.next;int newIndex=cur.key% tmpArray.length;//进行头插法cur.next=tmpArray[newIndex];tmpArray[newIndex]=cur;cur=newCur;}}array=tmpArray;}//进行取出public int get(int key){//判断k在哪里int index=key%array.length;Node cur=array[index];while (cur!=null){if(cur.key==key){return cur.val;}cur=cur.next;}return -1;}

这里小编模拟了哈希表开散列的放入数据的两种插入方式,即链表的头插法以及链表的尾插法。

注意扩容:

1.进行原来数值的项管部位置的插入,因为扩容后的数组容量大小进行变化,必须进行重新取余。

2.在进行在新的扩容数组后,使用的cur要进行保留,否则进行新的数组插入后,cur无法回到原来数组进行遍历剩余的数值遍历。 

 2.4性能分析

虽然哈希表一直在和冲突做斗争,但在实际使用过程中,我们认为哈希表的冲突率是不高的,冲突个数是可控的所以,通常意义下,我们认为哈希表的插入/删除/查找时间复杂度是O(1) 。

1.每个桶中的链表的长度是一个常数,并且可以进行调整。

2.负载因子的存在,使得在遍历时可以进数值过多的扩容。

2.5与Java类集的关系

1. HashMap 和 HashSet 即 java 中利用哈希表实现的 Map 和 Set


2. java 中使用的是哈希桶方式解决冲突的


3. java 会在冲突链表长度大于一定阈值后,将链表转变为搜索树(红黑树)


4. java 中计算哈希值实际上是调用的类的 hashCode 方法,进行 key 的相等性比较是调用 key 的 equals 方法。所以如果要用自定义类作为 HashMap 的 key 或者 HashSet 的值,必须覆写 hashCode 和 equals 方法。

 📚️3.总结

💬💬本期小编讲解了关于哈希表的内部原理,以及它的重点内部原理冲突的避免以及冲突的解决。

本期主要是解释性语言较多,注重理解,唯一的难点是开散列的模拟代码实现。

🌅🌅🌅~~~~最后希望与诸君共勉,共同进步!!!


                               💪💪💪以上就是本期内容了, 感兴趣的话,就关注小编吧。

                                                         😊😊  期待你的关注~~~

这篇关于【数据结构】关于哈希表内部原理,你到底了解多少???(超详解)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1123177

相关文章

springboot security快速使用示例详解

《springbootsecurity快速使用示例详解》:本文主要介绍springbootsecurity快速使用示例,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝... 目录创www.chinasem.cn建spring boot项目生成脚手架配置依赖接口示例代码项目结构启用s

Python中随机休眠技术原理与应用详解

《Python中随机休眠技术原理与应用详解》在编程中,让程序暂停执行特定时间是常见需求,当需要引入不确定性时,随机休眠就成为关键技巧,下面我们就来看看Python中随机休眠技术的具体实现与应用吧... 目录引言一、实现原理与基础方法1.1 核心函数解析1.2 基础实现模板1.3 整数版实现二、典型应用场景2

一文详解SpringBoot响应压缩功能的配置与优化

《一文详解SpringBoot响应压缩功能的配置与优化》SpringBoot的响应压缩功能基于智能协商机制,需同时满足很多条件,本文主要为大家详细介绍了SpringBoot响应压缩功能的配置与优化,需... 目录一、核心工作机制1.1 自动协商触发条件1.2 压缩处理流程二、配置方案详解2.1 基础YAML

Python实现无痛修改第三方库源码的方法详解

《Python实现无痛修改第三方库源码的方法详解》很多时候,我们下载的第三方库是不会有需求不满足的情况,但也有极少的情况,第三方库没有兼顾到需求,本文将介绍几个修改源码的操作,大家可以根据需求进行选择... 目录需求不符合模拟示例 1. 修改源文件2. 继承修改3. 猴子补丁4. 追踪局部变量需求不符合很

Java的IO模型、Netty原理解析

《Java的IO模型、Netty原理解析》Java的I/O是以流的方式进行数据输入输出的,Java的类库涉及很多领域的IO内容:标准的输入输出,文件的操作、网络上的数据传输流、字符串流、对象流等,这篇... 目录1.什么是IO2.同步与异步、阻塞与非阻塞3.三种IO模型BIO(blocking I/O)NI

java中反射(Reflection)机制举例详解

《java中反射(Reflection)机制举例详解》Java中的反射机制是指Java程序在运行期间可以获取到一个对象的全部信息,:本文主要介绍java中反射(Reflection)机制的相关资料... 目录一、什么是反射?二、反射的用途三、获取Class对象四、Class类型的对象使用场景1五、Class

golang 日志log与logrus示例详解

《golang日志log与logrus示例详解》log是Go语言标准库中一个简单的日志库,本文给大家介绍golang日志log与logrus示例详解,感兴趣的朋友一起看看吧... 目录一、Go 标准库 log 详解1. 功能特点2. 常用函数3. 示例代码4. 优势和局限二、第三方库 logrus 详解1.

一文详解如何从零构建Spring Boot Starter并实现整合

《一文详解如何从零构建SpringBootStarter并实现整合》SpringBoot是一个开源的Java基础框架,用于创建独立、生产级的基于Spring框架的应用程序,:本文主要介绍如何从... 目录一、Spring Boot Starter的核心价值二、Starter项目创建全流程2.1 项目初始化(

Spring Boot3虚拟线程的使用步骤详解

《SpringBoot3虚拟线程的使用步骤详解》虚拟线程是Java19中引入的一个新特性,旨在通过简化线程管理来提升应用程序的并发性能,:本文主要介绍SpringBoot3虚拟线程的使用步骤,... 目录问题根源分析解决方案验证验证实验实验1:未启用keep-alive实验2:启用keep-alive扩展建

Java异常架构Exception(异常)详解

《Java异常架构Exception(异常)详解》:本文主要介绍Java异常架构Exception(异常),具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1. Exception 类的概述Exception的分类2. 受检异常(Checked Exception)