0

0

c++怎么进行哈希表碰撞处理_c++ 链地址法与开放定址法实现【详解】

裘德小鎮的故事

裘德小鎮的故事

发布时间:2026-01-09 16:47:21

|

259人浏览过

|

来源于php中文网

原创

c++kquote>std::unordered_map默认用链地址法且不可替换为开放定址法;手写链地址法需桶数组+链表、质数容量与负载因子控制;开放定址法需删除标记和探测策略,二者适用场景不同。

c++怎么进行哈希表碰撞处理_c++ 链地址法与开放定址法实现【详解】

哈希表碰撞时,std::unordered_map 默认用链地址法,但你无法直接控制底层实现

标准库std::unordered_mapstd::unordered_set 确实基于哈希,且主流实现(如 libstdc++、libc++)内部采用链地址法(separate chaining),每个桶(bucket)挂一个链表或小动态数组来存冲突元素。但这是封装好的——你调用 insert()find() 时完全感知不到碰撞逻辑,也**不能替换为开放定址法**。想自定义碰撞处理?必须手写容器。

手写链地址法:用 std::vector + std::list 实现简易哈希表

核心是维护一个桶数组,每个桶存一个链表;哈希函数决定下标,冲突时往对应链表尾部插入。注意三点:

  • std::hash 可复用标准哈希,但需特化自定义类型(否则编译失败)
  • 桶数量应为质数,减少聚集;扩容时需重新哈希全部元素(rehash
  • 查找/删除需遍历链表,最坏 O(n),但平均 O(1) —— 前提是负载因子(size() / bucket_count())控制在 0.75 以下
template 
class SimpleHashMap {
    std::vector>> buckets;
    size_t num_elements = 0;
    static size_t next_prime(size_t n) { /* 返回大于 n 的最小质数 */ }

public: SimpleHashMap(size_t init_buckets = 11) : buckets(init_buckets) {}

void insert(const K& key, const V& val) {
    size_t idx = std::hashzuojiankuohaophpcnKyoujiankuohaophpcn{}(key) % buckets.size();
    for (auto& p : buckets[idx]) {
        if (p.first == key) { p.second = val; return; }
    }
    buckets[idx].emplace_back(key, val);
    ++num_elements;
    if (num_elements youjiankuohaophpcn buckets.size() * 0.75) rehash();
}

private: void rehash() { auto old = std::move(buckets); size_t new_size = next_prime(buckets.size() * 2); buckets.assign(new_size, std::list<:pair v>>{}); for (const auto& list : old) for (const auto& p : list) insert(p.first, p.second); } };

开放定址法难点在探测序列与删除标记,线性探测最易写但容易聚集

所有元素存于单一数组,碰撞时按固定规则(如线性、二次、双重哈希)找下一个空位。关键问题不是“怎么找”,而是:

  • 删除不能真删(否则断开探测链),得用特殊标记(如 EMPTYDELETED)区分“从未用过”和“曾用过已删”
  • 线性探测((hash + i) % capacity)实现简单,但连续冲突会形成“聚集”,性能骤降
  • 二次探测((hash + i*i) % capacity)缓解聚集,但可能找不到空位(即使有),需容量为质数且负载因子严格
  • 双重哈希((hash1 + i * hash2) % capacity)效果最好,但 hash2 必须与容量互质,实现稍复杂
template 
class OpenAddressingMap {
    struct Entry { K key; V val; enum { EMPTY, VALID, DELETED } state; };
    std::vector data;
    size_t num_valid = 0;
size_t probe(size_t hash, size_t i) const {
    return (hash + i * i) % data.size(); // 二次探测
}

public: V& operator[](const K& key) { size_t h = std::hash{}(key); for (size_t i = 0; i

选链地址法还是开放定址法?看场景,别只看理论平均复杂度

链地址法内存开销略大(指针 + 动态分配),但实现鲁棒、支持任意负载因子、删除即释放;开放定址法缓存友好(数据连续)、无指针开销,但扩容成本高、删除麻烦、对哈希质量更敏感。实际选型时:

堆友
堆友

Alibaba Design打造的设计师全成长周期服务平台,旨在成为设计师的好朋友

下载

立即学习C++免费学习笔记(深入)”;

  • 元素小(如 int 键值对)、读多写少、内存敏感 → 开放定址法更合适
  • 键类型复杂、需频繁增删、不希望手动管理探测逻辑 → 链地址法更省心
  • std::unordered_map 就别纠结——它已针对通用场景优化过桶结构(如 libc++ 用小数组代替链表,减少分配)
  • 真正要极致性能?别自己写,用 absl::flat_hash_map(开放定址+混合策略)或 tsl::robin_map

手写开放定址法时,DELETED 标记状态和探测终止条件最容易漏判,一错就导致查不到或无限循环。

相关专题

更多
if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

723

2023.08.22

c语言const用法
c语言const用法

const是关键字,可以用于声明常量、函数参数中的const修饰符、const修饰函数返回值、const修饰指针。详细介绍:1、声明常量,const关键字可用于声明常量,常量的值在程序运行期间不可修改,常量可以是基本数据类型,如整数、浮点数、字符等,也可是自定义的数据类型;2、函数参数中的const修饰符,const关键字可用于函数的参数中,表示该参数在函数内部不可修改等等。

520

2023.09.20

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

315

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

532

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

51

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

193

2025.08.29

javascriptvoid(o)怎么解决
javascriptvoid(o)怎么解决

javascriptvoid(o)的解决办法:1、检查语法错误;2、确保正确的执行环境;3、检查其他代码的冲突;4、使用事件委托;5、使用其他绑定方式;6、检查外部资源等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

174

2023.11.23

java中void的含义
java中void的含义

本专题整合了Java中void的相关内容,阅读专题下面的文章了解更多详细内容。

95

2025.11.27

c++主流开发框架汇总
c++主流开发框架汇总

本专题整合了c++开发框架推荐,阅读专题下面的文章了解更多详细内容。

3

2026.01.09

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
10分钟--Midjourney创作自己的漫画
10分钟--Midjourney创作自己的漫画

共1课时 | 0.1万人学习

Midjourney 关键词系列整合
Midjourney 关键词系列整合

共13课时 | 0.9万人学习

AI绘画教程
AI绘画教程

共2课时 | 0.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号