news 2026/2/5 8:58:41

代码随想录算法训练营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/1/29 13:27:59

5.2 MCP架构角色:深入理解Client与Server交互模式

5.2 MCP架构角色:深入理解Client与Server交互模式 在上一节中,我们介绍了MCP协议的基本概念和核心机制。本节将深入探讨MCP架构中Client和Server的角色分工,详细分析它们之间的交互模式,以及如何构建高效、安全的MCP系统。 MCP架构概览 MCP采用客户端-服务器架构,其中C…

作者头像 李华
网站建设 2026/1/30 19:14:40

知名Java开发工具IntelliJ IDEA v2025.3正式上线——开发效率全面提升

IntelliJ IDEA 是由 JetBrains 开发的智能 Java IDE&#xff0c;提供代码自动补全、重构工具、框架集成&#xff08;Spring/JPA 等&#xff09;、数据库工具和调试支持&#xff0c;通过深度代码分析与跨语言功能优化企业级开发流程&#xff0c;被广泛认可为专业 Java 开发者的高…

作者头像 李华
网站建设 2026/2/3 3:28:57

高并发系统性能测试中的用户数测算体系研究

在数字化业务爆发式增长的2025年&#xff0c;电商平台、金融服务、在线教育等领域的系统频繁面临高并发访问压力。准确的并发用户数测算成为性能测试成功的核心前提&#xff0c;直接关系到系统稳定性评估与容量规划精确度。本文基于软件工程理论与行业实践&#xff0c;构建涵盖…

作者头像 李华
网站建设 2026/2/3 16:28:10

G-Star 精选开源项目推荐|12.7 — 12.12

G-Star 开源摘星计划&#xff0c;简称 G-Star 计划&#xff0c;是 AtomGit 平台推出的针对开源项目成长全流程的扶持计划&#xff0c;我们为每一个申请加入 G-Star 计划的开源项目提供资源对接与运营支持&#xff1a;包括代码托管、品牌市场推广、社区化运营等。参与 G-Star 计…

作者头像 李华
网站建设 2026/2/4 21:19:30

模数转换芯片FZH709,应用开发相关数据技术手册

1 芯片功能说明 FZH709 是一款高精度、低功耗模数转换芯片&#xff0c;一路差分输入通道&#xff0c;内置温度传感器和 高精度振荡器。 FZH709 的 PGA 可选&#xff1a;1、2、64、128&#xff0c;默认为 128。 FZH709 正常模式下的 ADC 数据输出速率可选&#xff1a;10Hz、40Hz…

作者头像 李华