前言
前面我们使用开放定址法实现了哈希表。
对于开放定址法来说,包含以下两种探测插入节点位置方法:
- 线性探测
- 二次探测

但是开放定址法的缺点也很明显,开放定址法容易很多数据堆积在一起,大大减少了效率。
为了解决上述问题,引入了第二种方法实现哈希表——链地址法(哈希桶法)
一、链地址法概念
开放定址法中,所有的元素都放到哈希表里。
链地址法中,所有的数据不再直接存储在哈希表中。哈希表中存储一个指针,没有数据映射到这个位置时,这个指针为空;有多个数据映射到这个位置时,我们把这些冲突的数据链接成一个链表,挂在哈希表这个位置下面。链地址法也叫做拉链法或者哈希桶。
下⾯演⽰ {19,30,5,36,13,20,21,12,24,96} 等这⼀组值映射到 M=11 的表中。

二、哈希表扩容
开放定址法的负载因子必须小于 1,而链地址法的负载因子则没有限制,可以大于 1。
负载因子越大,哈希冲突的概率越高,空间利用率越高;负载因子越小,哈希冲突的概率越低,空间利用率越低。
STL 中 unordered_xxx 的最大负载因子基本控制在 1,当负载因子大于 1 时会扩容。我们下面的实现也使用这种方式。
也就是说,我们期望基本每个节点下面都挂一个桶,有那么一两个数据,如下图:

三、哈希桶插入逻辑
首先,如果不需要扩容,我们需要将一个节点挂上去,因为每一个哈希桶类似于链表,而链表的头插效率是十分高的,因此我们采用头插。

// 如果不需要扩容
size_t hashi = hf(kv.first) % _table.size();
// 头插
Node* newnode = new Node(kv);
newnode->_next = _table[hashi];
_table[hashi] = newnode;
++_n;
;





