|
|
|
@ -1,10 +1,10 @@
|
|
|
|
|
# 哈希表
|
|
|
|
|
|
|
|
|
|
哈希表通过建立「键 key」与「值 value」之间的映射,实现高效的元素查询。具体而言,我们向哈希表输入一个 key,则可以在 $O(1)$ 时间内获取对应的 value 。
|
|
|
|
|
「哈希表 Hash Table」通过建立键 `key` 与值 `value` 之间的映射,实现高效的元素查询。具体而言,我们向哈希表输入一个 `key` ,则可以在 $O(1)$ 时间内获取对应的 `value` 。
|
|
|
|
|
|
|
|
|
|
以一个包含 $n$ 个学生的数据库为例,每个学生都有“姓名 `name`”和“学号 `id`”两项数据。假如我们希望实现查询功能,例如“输入一个学号,返回对应的姓名”,则可以采用哈希表来实现。
|
|
|
|
|
|
|
|
|
|
![哈希表的抽象表示](hash_map.assets/hash_map.png)
|
|
|
|
|
![哈希表的抽象表示](hash_map.assets/hash_table_lookup.png)
|
|
|
|
|
|
|
|
|
|
除哈希表外,我们还可以使用数组或链表实现元素查询,其中:
|
|
|
|
|
|
|
|
|
@ -12,8 +12,6 @@
|
|
|
|
|
- 添加元素仅需添加至尾部即可,使用 $O(1)$ 时间;
|
|
|
|
|
- 删除元素需要先查询再删除,使用 $O(n)$ 时间;
|
|
|
|
|
|
|
|
|
|
然而,在哈希表中进行增删查的时间复杂度都是 $O(1)$ 。哈希表全面胜出!因此,哈希表常用于对查找效率要求较高的场景。
|
|
|
|
|
|
|
|
|
|
<div class="center-table" markdown>
|
|
|
|
|
|
|
|
|
|
| | 数组 | 链表 | 哈希表 |
|
|
|
|
@ -24,6 +22,8 @@
|
|
|
|
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
观察发现,在哈希表中进行增删查改的时间复杂度都是 $O(1)$ ,非常高效。因此,哈希表常用于对查找效率要求较高的场景。
|
|
|
|
|
|
|
|
|
|
## 哈希表常用操作
|
|
|
|
|
|
|
|
|
|
哈希表的基本操作包括 **初始化、查询操作、添加与删除键值对**。
|
|
|
|
@ -428,29 +428,26 @@
|
|
|
|
|
});
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
## 哈希函数
|
|
|
|
|
|
|
|
|
|
哈希表的底层实现为数组,同时可能包含链表、二叉树(红黑树)等数据结构,以提高查询性能(将在下节讨论)。
|
|
|
|
|
|
|
|
|
|
首先考虑最简单的情况,**仅使用一个数组来实现哈希表**。通常,我们将数组中的每个空位称为「桶 Bucket」,用于存储键值对。
|
|
|
|
|
|
|
|
|
|
我们将键值对 key, value 封装成一个类 `Pair` ,并将所有 `Pair` 放入数组中。这样,数组中的每个 `Pair` 都具有唯一的索引。为了建立 key 和索引之间的映射关系,我们需要使用「哈希函数 Hash Function」。
|
|
|
|
|
## 哈希表简单实现
|
|
|
|
|
|
|
|
|
|
设哈希表的数组为 `buckets` ,哈希函数为 `f(x)` ,那么查询操作的步骤如下:
|
|
|
|
|
我们先考虑最简单的情况,**仅用一个数组来实现哈希表**。在哈希表中,我们通常将数组中的每个空位称为「桶 Bucket」,每个桶可存储一个键值对。因此,查询操作就是定位输入的 `key` 对应的桶,从而得到 `value` 。
|
|
|
|
|
|
|
|
|
|
1. 输入 `key` ,通过哈希函数计算出索引 `index` ,即 `index = f(key)` ;
|
|
|
|
|
2. 通过索引在数组中访问到键值对 `pair` ,即 `pair = buckets[index]` ,然后从 `pair` 中获取对应的 `value` ;
|
|
|
|
|
那么,如何基于 `key` 来定位对应的桶呢?这是通过「哈希函数 Hash Function」实现的。哈希函数的作用是将一个较大的输入空间映射到一个较小的输出空间。在哈希表中,输入空间是所有 `key` ,输出空间是所有桶(数组索引)。换句话说,**输入一个 `key` ,我们可以通过哈希函数得到该 `key` 对应的键值对存储在数组中的位置**。
|
|
|
|
|
|
|
|
|
|
以学生数据 `key 学号 -> value 姓名` 为例,我们可以设计如下哈希函数:
|
|
|
|
|
哈希函数的计算过程分为两步:输入一个 `key` ,首先通过函数 `hash(key)` 计算得到哈希值,接下来将哈希值对桶数量(数组长度)取模,从而获取该 `key` 对应的数组索引 `index` 。计算公式如下
|
|
|
|
|
|
|
|
|
|
$$
|
|
|
|
|
f(x) = x \bmod {100}
|
|
|
|
|
index = \text{hash}(key) \bmod {c}
|
|
|
|
|
$$
|
|
|
|
|
|
|
|
|
|
其中 $\bmod$ 表示取余运算。
|
|
|
|
|
其中, $\bmod$ 表示取余运算, $c$ 为桶数量(数组长度)。随后,我们就可以利用 `index` 在哈希表中访问对应的桶,从而获取 `value` 。
|
|
|
|
|
|
|
|
|
|
设数组长度 $c = 100$ , $\text{hash}(key) = key$ ,易得哈希函数为 $key \bmod 100$ 。下图以 `key` 学号和 `value` 姓名为例,展示了哈希函数的工作原理。
|
|
|
|
|
|
|
|
|
|
![哈希函数工作原理](hash_map.assets/hash_function.png)
|
|
|
|
|
|
|
|
|
|
以下代码给出了一个简单哈希表实现。其中,我们将 `key` 和 `value` 封装成一个类 `Pair` ,以表示键值对。
|
|
|
|
|
|
|
|
|
|
=== "Java"
|
|
|
|
|
|
|
|
|
|
```java title="array_hash_map.java"
|
|
|
|
@ -541,18 +538,14 @@ $$
|
|
|
|
|
|
|
|
|
|
## 哈希冲突
|
|
|
|
|
|
|
|
|
|
细心的你可能已经注意到,**在某些情况下,哈希函数 $f(x) = x \bmod 100$ 可能无法正常工作**。具体来说,当输入的 key 后两位相同时,哈希函数的计算结果也会相同,从而指向同一个 value 。例如,查询学号为 $12836$ 和 $20336$ 的两个学生时,我们得到:
|
|
|
|
|
本质上看,哈希函数的是将一个庞大的输入空间(`key` 范围)映射到一个较小的输出空间(数组索引范围)。因此,**理论上一定存在”多个输入对应相同输出”的情况**。
|
|
|
|
|
|
|
|
|
|
对于上述示例中的哈希函数,当输入的 `key` 后两位相同时,哈希函数的输出结果也相同。例如,查询学号为 $12836$ 和 $20336$ 的两个学生时,我们得到:
|
|
|
|
|
|
|
|
|
|
$$
|
|
|
|
|
f(12836) = f(20336) = 36
|
|
|
|
|
12836 \bmod 100 = 20336 \bmod 100 = 36
|
|
|
|
|
$$
|
|
|
|
|
|
|
|
|
|
这两个学号指向了同一个姓名,这显然是错误的。我们把这种情况称为「哈希冲突 Hash Collision」。在后续章节中,我们将讨论如何解决哈希冲突的问题。
|
|
|
|
|
两个学号指向了同一个姓名,这显然是不对的。我们把这种情况称为“哈希冲突”。在下节中,我们将重点讨论如何解决冲突问题。
|
|
|
|
|
|
|
|
|
|
![哈希冲突示例](hash_map.assets/hash_collision.png)
|
|
|
|
|
|
|
|
|
|
综上所述,一个优秀的哈希函数应具备以下特性:
|
|
|
|
|
|
|
|
|
|
- 尽可能减少哈希冲突的发生;
|
|
|
|
|
- 查询效率高且稳定,能够在绝大多数情况下达到 $O(1)$ 时间复杂度;
|
|
|
|
|
- 较高的空间利用率,即使“键值对占用空间 / 哈希表总占用空间”比例最大化;
|
|
|
|
|