news 2026/6/23 22:45:18

代码随想录 109.冗余连接Ⅱ

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
代码随想录 109.冗余连接Ⅱ

一、思路:

(1)本题和684.冗余连接类似,但本题是一个有向图,相对要复杂一些。

(2)题目要求:有一个有向图,是由一棵有向树 + 一条有向边组成的(所以此时这个图就不能称之为有向树),现在要求找到这条有向边并将其删除,使得这个图可以恢复为有向树。

(3)若有多条可以删除的边,则删除顺序靠后的边。

(4)有向树的性质:对于一棵有向树,只有根节点的入度为0,其他节点的入度都为1(因为该树除了根节点之外的每一个节点都有且只有一个父节点,而根节点没有父节点)。

二、分情况讨论:

1.情况一:找到入度为2的节点,则直接删除一条指向该节点的边即可。

如下图所示:

找到了节点3的入度为2,删1->3或者2->3,选择删顺序靠后的便可。

2.情况二:入度为2的节点还有一种情况,如下图所示:

节点3的入度为2,但在删除边的时候,由于存在有向环,只能删这条边(节点1->节点3),如果删除边(节点4->节点3),那么删除后本图也不是有向树了,因为找不到根节点。

综上所述,如果遇到入度为2的节点,需要判断删哪一条边,删除后能使本图成为有向树。如果是删哪个都可以,优先删顺序靠后的边。

3.情况三:如果没有入度为2的点,说明图中有环(注意是有向环)且添加的边指向了根节点,如下图所示:

此时删掉构成环的边即可。

三、解决方法。

1.首先计算每个节点的入度,如果存在入度为2的节点,就定位到该节点对应的两条边,分别记为dup[0]和dup[1]。如果在删除dup[1]后,剩余的边无法形成树,说明dup[0]是需要删除的边;否则说明dup[1]是需要删除的边。

2.如果不存在入度为2的节点,就遍历数组edges,对于每条边(u,v),使用并查集维护节点之间的连通性。如果u和v已经连通,说明图中存在有向环,此时当前边即为需要删除的边。

附代码:

class Solution { int[] parent; int find(int x) { if (parent[x] != x) parent[x] = find(parent[x]); return parent[x]; } void union(int x, int y) { if (find(x) != find(y)) parent[find(y)] = parent[x]; } public int[] findRedundantDirectedConnection(int[][] edges) { parent = new int[1001]; int[] in = new int[1001]; //记录每个节点的入度 int[] res = {}; //存储可能的多余边 // 寻找是否存在入度为 2 的顶点 for (int[] e : edges) { if (++in[e[1]] == 2) { //如果某个节点的入度变为2 res = e; //记录最后一条使入度变为2的边 } } // 如果存在入度为 2 的顶点 // 尝试删除指向该顶点的某一条边,看剩下的点是否能够构成树 // 如果可以构成树,直接返回该边,否则返回另一条边 if (res.length != 0) { if (check(edges, res)) return res; //如果删除res边后能成树,则返回res else { //否则返回另一条指向同一个节点的边 for (int[] e : edges) if (e[1] == res[1]) return e; } } // 重新初始化并查集 for (int i = 0; i < 1001; i++) { parent[i] = i; } for (int[] e : edges) { //如果两个节点已经在同一个集合,说明这条边会成环 // 删除加入形成环的边 if (find(e[0]) == find(e[1])) return e; else union(e[0], e[1]); } return new int[0]; } // 判断有向边构成的图形是否为树 boolean check(int[][] edges, int[] remove) { // 初始化并查集 for (int i = 0; i < 1001; i++) { parent[i] = i; } for (int[] e : edges) { // 跳过要删除的边 if (Arrays.equals(e, remove)) continue; // 如果删除后还有环,说明删除的不是正确的边 if (find(e[0]) == find(e[1])) return false; else union(e[0], e[1]); } return true; } }
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/6/23 8:27:19

1小时搭建DHT11物联网监控原型

快速体验 打开 InsCode(快马)平台 https://www.inscode.net输入框内输入如下内容&#xff1a; 快速生成一个DHT11物联网监控系统的最小可行原型。要求&#xff1a;1. NodeMCU连接DHT11 2. 数据通过MQTT上传到云平台 3. 简单的Web界面实时展示数据 4. 手机端适配 5. 一键部署脚…

作者头像 李华
网站建设 2026/6/23 18:23:30

Gazebo仿真入门:零基础搭建第一个机器人世界

快速体验 打开 InsCode(快马)平台 https://www.inscode.net输入框内输入如下内容&#xff1a; 开发一个交互式Gazebo学习助手&#xff0c;功能包括&#xff1a;1) 分步指导安装和配置Gazebo 2) 可视化界面创建简单机器人模型(如小车) 3) 拖拽式场景搭建 4) 基础物理属性调整演…

作者头像 李华
网站建设 2026/6/23 18:19:57

零基础玩转DHT11:从接线到数据读取全指南

快速体验 打开 InsCode(快马)平台 https://www.inscode.net输入框内输入如下内容&#xff1a; 创建一个面向初学者的DHT11教学项目&#xff0c;包含&#xff1a;1. 清晰的硬件接线示意图 2. 最简单的Arduino示例代码 3. 常见问题解答 4. 数据读取结果串口打印示例 5. 基础电路…

作者头像 李华
网站建设 2026/6/23 4:10:50

传统网络配置 vs AI辅助:处理10.8.8.8的效率对比

快速体验 打开 InsCode(快马)平台 https://www.inscode.net输入框内输入如下内容&#xff1a; 构建一个网络配置效率对比工具&#xff0c;分别展示手动配置10.8.8.8等IP的步骤和耗时&#xff0c;与AI自动化配置的流程对比。包含时间统计、错误率分析和配置复杂度评估。使用可视…

作者头像 李华
网站建设 2026/6/23 7:58:27

如何用AI自动修复代理连接错误?快马平台实战

快速体验 打开 InsCode(快马)平台 https://www.inscode.net输入框内输入如下内容&#xff1a; 创建一个能自动诊断和修复ERR_PROXY_CONNECTION_FAILED错误的工具。功能包括&#xff1a;1. 自动检测系统代理设置 2. 分析网络连接状态 3. 提供修复建议 4. 一键应用修复方案 5. 生…

作者头像 李华
网站建设 2026/6/23 18:18:44

传统vsAI:全球项目交付速度提升300%的秘诀

快速体验 打开 InsCode(快马)平台 https://www.inscode.net输入框内输入如下内容&#xff1a; 构建一个项目效率对比工具&#xff0c;能够并行运行传统开发流程和AI辅助流程&#xff0c;实时显示两者在代码生成、测试、部署等环节的时间差异和产出质量对比。工具应支持自定义项…

作者头像 李华