0

0

C++怎么实现并查集_C++集合合并教程【连通】

穿越時空

穿越時空

发布时间:2026-02-21 13:09:12

|

579人浏览过

|

来源于php中文网

原创

并查集必须同时实现路径压缩和按秩合并才能保证均摊o(α(n));find需更新父指针,union需用rank比较而非值大小,初始化不可遗漏rank数组,离散化或哈希映射可处理非连续编号。

c++怎么实现并查集_c++集合合并教程【连通】

怎么写一个靠谱的 unionfind

并查集不是靠“教程”堆出来的,是靠两个函数能不能扛住路径压缩和按秩合并的双重压力。手写时最容易出问题的是 find 没做路径压缩,或者 union 没比较秩(rank)直接乱连,导致树退化成链表,find 从 O(α(n)) 变成 O(n)。

实操建议:

  • find 必须递归或迭代实现路径压缩:找到根后,把当前节点父指针直接指向根,别只返回根而不改结构
  • union 别用值大小比较,用 rank 数组(或 size,但得统一)决定谁当爹;rank 相等时才给一方 rank++
  • 初始化时,每个 parent[i] = irank[i] = 0,别漏掉 rank 数组初始化
  • 如果用 vector<int></int>parentrank,下标从 0 开始,别拿输入节点编号直接当索引——比如节点编号是 1~n,就开 vector<int>(n+1)</int>

为什么 std::setstd::unordered_set 不能替代并查集

因为它们不维护集合间的连通关系。你往 std::set 里插一堆数,它只管去重和排序;哪怕两个集合元素完全一样,你也看不出它们是否该被“合并”。并查集的核心是动态维护等价类,而标准容器没提供 mergeconnected 这类语义操作。

常见错误现象:

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

AMiner
AMiner

AMiner——新一代智能型科技情报挖掘与服务系统,能够为你提供查找论文、理解论文、分析论文、写作论文四位一体一站式服务。

下载
  • 试图用多个 std::set 手动合并——每次 insert 大量元素,时间复杂度飙升,且无法快速回答“a 和 b 是否同属一集”
  • std::map<int int></int> 存 parent 但忘了写 find 压缩,查几次就栈溢出或超时
  • 误以为 std::disjoint_set 是标准库组件——C++23 还没进,别搜了,没有

路径压缩 + 按秩合并,哪个能省?

都不能省。单独用路径压缩,最坏情况 union 会拉高树高;单独用按秩合并,find 仍是 O(log n)。两者合体才能保证单次操作均摊 O(α(n)),α 是反阿克曼函数,对所有实际输入都 ≤ 4。

实操注意点:

  • 按秩合并中的“秩”不是真实树高,只是上界估计,所以可以安全地只在两树秩相等时才 rank[root]++
  • 路径压缩后,rank 值会失真,但这没关系——它只用于 union 决策,不参与 find 计算
  • 如果图是静态的、只查询不修改,考虑用 DFS/BFS 求连通分量更直白;并查集的价值在“边加边查”的在线场景

边界情况:节点编号不连续 or 有负数怎么办

并查集底层是数组索引映射,不支持负数或稀疏编号。硬塞会越界或浪费内存。

解决办法很简单:

  • 离散化:把所有出现过的节点值扔进 vector,排序去重,用 lower_bound 映射到 0~n-1
  • unordered_map<int int></int> 代替 vectorparent,但要注意:find 里每次访问都要查哈希表,常数变大;而且你得自己处理未注册节点——第一次 find(-5) 时,得先 parent[-5] = -5
  • 别用 map(红黑树),查找慢还容易写错默认构造逻辑

离散化比哈希映射更稳,尤其数据量不大时;真要支持任意整数且动态极强,哈希方案就得配好初始化逻辑,否则第一次 find 就崩。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
c语言union的用法
c语言union的用法

c语言union的用法是一种特殊的数据类型,它允许在相同的内存位置存储不同的数据类型,union的使用可以帮助我们节省内存空间,并且可以方便地在不同的数据类型之间进行转换。使用union时需要注意对应的成员是有效的,并且只能同时访问一个成员。本专题为大家提供union相关的文章、下载、课程内容,供大家免费下载体验。

128

2023.09.27

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

421

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

594

2023.08.10

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

421

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

594

2023.08.10

golang map内存释放
golang map内存释放

本专题整合了golang map内存相关教程,阅读专题下面的文章了解更多相关内容。

77

2025.09.05

golang map相关教程
golang map相关教程

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

36

2025.11.16

golang map原理
golang map原理

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

67

2025.11.17

pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法
pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法

本专题系统整理pixiv网页版官网入口及登录访问方式,涵盖官网登录页面直达路径、在线阅读入口及快速进入方法说明,帮助用户高效找到pixiv官方网站,实现便捷、安全的网页端浏览与账号登录体验。

796

2026.02.13

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
C# 教程
C# 教程

共94课时 | 9.8万人学习

C 教程
C 教程

共75课时 | 4.9万人学习

C++教程
C++教程

共115课时 | 18.6万人学习

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

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