news 2026/6/2 11:39:44

代码随想录算法训练营Day47 | 并查集理论基础、107.寻找存在的路线

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
代码随想录算法训练营Day47 | 并查集理论基础、107.寻找存在的路线

并查集理论基础

一、核心思想

高效处理动态连通性问题。

并查集用于判断两个元素是否在同一个集合中。它将每个集合看作一棵树,集合的“代表”就是这棵树的根节点。如果两个元素的根节点相同,它们就在同一个集合。

二、三大核心操作
  1. 初始化
    • 功能:开始时,每个元素都是一个独立的集合,其根节点是自己。
    • 代码:
void init(int n) { for (int i = 0; i < n; ++i) { father[i] = i; // 每个节点的父节点都是自己 } }
  1. 寻根
    • 功能:找到指定元素所在集合的根节点。这是并查集的灵魂。
    • 优化(路径压缩):在寻找根的过程中,将路径上的所有节点直接指向根节点,极大提升后续查询效率。
    • 代码:
int find(int u) { // 如果u的父节点不是自己,就递归寻找,并把u的父节点直接设置为根 return u == father[u] ? u : father[u] = find(father[u]); }
  1. 合并
    • 功能:将两个元素所在的集合合并成一个。
    • 核心原则:必须先找到两个元素的根节点,再将其中一个根节点连接到另一个上。
    • 代码:
void join(int u, int v) { u = find(u); // 找到u的根 v = find(v); // 找到v的根 if (u == v) return; // 如果根相同,说明已在同一集合,无需操作 father[v] = u; // 将v的根连接到u的根上 }
  1. 判断
    • 功能:判断两个元素是否在同一个集合。
    • 代码:
bool isSame(int u, int v) { return find(u) == find(v); }
三、常见误区:join函数的正确写法

错误写法:

void join(int u, int v) { if (isSame(u, v)) return; // 虽然判断对了,但... father[v] = u; // ...这里连接的是原始节点u和v,而不是它们的根! }
  • 问题:这样会破坏树的结构,导致后续find操作出错。例如,join(1, 2); join(3, 2);后,isSame(1, 3)会返回false,不符合预期。

正确写法:

void join(int u, int v) { u = find(u); // 必须先找根! v = find(v); // 必须先找根! if (u == v) return; father[v] = u; // 连接的是根节点 }
  • 原因:保证了总是将两个集合的根进行连接,维护了数据结构的正确性。
四、另一个优化:按秩合并
  • 思想:在合并时,总是将“秩”(可以理解为树的高度或大小)较小的树挂载到较大的树上,避免树的高度过快增长。
  • 虽然理论上很好,但路径压缩的优化效果已经非常出色,且代码更简洁。在实际应用和面试中,通常只使用路径压缩就足够了。
五、完整代码模板
int n = 1005; // n根据题目中节点数量而定,一般比节点数量大一点就好 vector<int> father = vector<int> (n, 0); // 并查集初始化 void init() { for (int i = 0; i < n; ++i) { father[i] = i; } } // 并查集里寻根的过程 int find(int u) { return u == father[u] ? u : father[u] = find(father[u]); // 路径压缩 } // 判断 u 和 v是否找到同一个根 bool isSame(int u, int v) { u = find(u); v = find(v); return u == v; } // 将v->u 这条边加入并查集 void join(int u, int v) { u = find(u); // 寻找u的根 v = find(v); // 寻找v的根 if (u == v) return ; // 如果发现根相同,则说明在一个集合,不用两个节点相连直接返回 father[v] = u; }
六、复杂度分析
  • 空间复杂度:O(n),需要一个father数组。
  • 时间复杂度:接近 O(1)。
    • 路径压缩后的并查集时间复杂度在O(logn)与O(1)之间,且随着查询或者合并操作的增加,时间复杂度会越来越趋于O(1)。
    • 路径压缩保证了每次操作后,树的结构都趋向于扁平化,使得后续的查询和合并操作非常快。

KamaCoder107.寻找存在的路线

107. 寻找存在的路线

1.思路

本题是并查集的模板题,掌握基础模板就能直接拿下。

#include <iostream> #include <vector> using namespace std; int n,m; vector<int>father(105,1); void init(){ for(int i=1;i<=n;i++){ father[i]=i; } } int find(int u){ if(u==father[u]) return u; return father[u]=find(father[u]); } bool issame(int u,int v){ u=find(u); v=find(v); return u==v; } void join(int u,int v){ u=find(u); v=find(v); if(u==v) return; father[u]=v; } int main(){ cin>>n>>m; init(); for(int i=0;i<m;i++){ int s,t;cin>>s>>t; join(s,t); } int source,destination; cin>>source>>destination; cout<<issame(source,destination); return 0; }

2.Reference:107. 寻找存在的路径 | 代码随想录

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/5/31 11:13:55

Claude-Opus-4.5国内接入实战:从网络层到业务层的工程化落地

在Claude-Opus-4.5发布后&#xff0c;在复杂推理场景下的表现引起了开发界的广泛关注。但对于国内团队而言&#xff0c;如何将稳定集成到生产环境&#xff0c;面临着网络、支付与 SDK 兼容这“三座大山”。本文基于实际开发经验&#xff0c;拆解接入难点并分享了一套其高可用的…

作者头像 李华
网站建设 2026/5/31 21:45:02

5.2 MCP协议详解:核心机制与设计思想

5.2 MCP协议详解:核心机制与设计思想 在上一节课中,我们深入探讨了LLM的两个致命痛点:上下文窗口限制和知识滞后性。为了解决这些问题,业界提出了Model Context Protocol (MCP)这一创新性解决方案。本节课将详细介绍MCP协议的核心机制与设计思想,帮助我们理解如何通过这一…

作者头像 李华
网站建设 2026/5/31 21:44:46

前端场景题,零基础入门到精通,收藏这篇就够了

前言 ​ 2026年的春招聘还有两个月就即将到来&#xff0c;为了帮助前端求职者提升复习效率&#xff0c;更快的拿到前端offer ​ 所以&#xff0c;我咨询了超过18位资深中大厂面试官后&#xff0c;准确精炼了一套切实可行的场景题&#xff0c;现在已经有432位粉丝通过这套题走…

作者头像 李华
网站建设 2026/5/31 20:54:13

写英文论文可不可以引用中文文献?

很多写SCI论文的同学经常会遇到一个问题&#xff1a;自己写的英文论文&#xff0c;不仅涉及到外文文献&#xff0c;同时也涉及到中文论文&#xff0c;是否可以引用中文参考文献呢&#xff1f; 答案是可以的。 中文文献也属于国际论文中重要参考文献的一部分&#xff0c;如果写…

作者头像 李华
网站建设 2026/6/1 15:27:21

2025大模型效率革命:Qwen3双模式切换重塑企业AI应用范式

2025大模型效率革命&#xff1a;Qwen3双模式切换重塑企业AI应用范式 【免费下载链接】Qwen3-32B-MLX-8bit 项目地址: https://ai.gitcode.com/hf_mirrors/Qwen/Qwen3-32B-MLX-8bit 导语 阿里通义千问Qwen3系列模型以创新的单模型双模式切换技术&#xff0c;重新定义大…

作者头像 李华