zoukankan      html  css  js  c++  java
  • 记录一下哈希表底层原理

    理解HashMap底层,首先应该理解Hash函数

    从解决一个问题入手:大量的数据要存储查询,构造哈希表来解决

    初步想法

    借鉴数组下标访问的思路来做,只需知道起始位置和下标值,

    不管数组中有多少个元素,都可以一次访问到,

    将元素和元素位置建立一种一一对应的关系

    Hash函数的出现

    输入的元素的范围可能很大甚至无穷,而我们的内存有限,

    所以说我们需要一种函数映射关系,将这些无限的元素映射到我们有限的内存地址上。

    Hash函数代表着一类函数,即把任意范围的元素可以通过映射关系压缩成固定范围的元素。

    Hash函数的选择

    如果是正整数,我们可以用这个正整数数除以某个数,取其余数,即我们常用的 k % m,k为正整数,m 为除数;

    这样一来,范围就缩小了很多,比如说 15%10=5,26%10=6,...,所有的正整数经过运算,都变成了 0-9 范围之间的数了,

    这样范围就缩小了很多

    m 的选择

    这种做法,m 的选择就非常重要了,如果 k 值分布均匀还无所谓,如果 k 值具有某些特征

    比如说 k 的个位基本上不变,而高位分布均匀,如 15,25,45,65,85,95,155,就遭遇大冲突了,

    必须要使得经过Hash函数后关键字的分布均匀,尽量减少冲突

    链地址法

    为了解决冲突,引出链地址法

    在存储的时候,如果多个元素被Hash到同一位置,那么就加入到该位置所指向的链表中,

    如果该位置没有元素,则为null(指向空)”

    由于新加入的元素很可能被再次访问到,使用“头插”

    rehash

    这样解决冲突固然好,但是也有瓶颈

    当我们实际存入的值越来越多的时候,这个链表也势必越来越长,

    那当我们进行查找的时候,势必就会遍历链表,效率也就越来越慢。

    因此,我们要选取一个相关的新的Hash函数(比如之前使用 key % m,现在只改变一下m的值)

    将旧Hash表中所有的元素通过新的Hash函数计算出新的Hash值,并将其插入到新表中(仍然使用链表),这就叫rehash

     这里的数组就扩大了近两倍,由于要大小要选素数,那就选原数组大小两倍后的第一个素数7,旧Hash表和新Hash表采用了不同的Hash函数,但相关,只是m的取值变了

    装载因子 α 

     我们可以定义这样一个变量 α = 所有元素个数/数组的大小,

    它代表着我们的Hash表(也就是数组)的装满程度,在这里也代表链表的平均长度

     这个装载因子代表了Hash表的装满程度,这里也可以代表链表的平均长度,那么也就可以代表查询时的时间长短了。

    参考资料:神速哈希上神速哈希下

  • 相关阅读:
    集训作业 洛谷P1135 奇怪的电梯
    集训作业 洛谷P1433 吃奶酪
    集训作业 洛谷P1443 马的遍历
    集训作业 洛谷P1032 字串变换
    T3 成绩单 题解
    T4 字符串的修改 题解
    P3379 最近公共祖先(LCA) 洛谷
    集训作业 洛谷P3913 车的攻击
    解决intellij idea卡顿的方法
    好的在线开发工具推荐
  • 原文地址:https://www.cnblogs.com/kumata/p/9857341.html
Copyright © 2011-2022 走看看