【代码随想录训练营第42期 Day56打卡 - 图论Part6 - 并查集2 - 冗余连接问题

目录

一、做题心得

二、题目与题解

题目一:108. 冗余连接

题目链接

题解:并查集

题目二:109. 冗余连接II

题目链接

题解:并查集

三、小结


一、做题心得

冗杂连接问题是图论章节应用并查集的经典问题。所有的顶点通过边相连,且除了一个多余的边之外,其他的边都是必须的,用以保持图的所有顶点连通,而我们需要做的,就是删除这条边,即冗杂的边。

这里打卡的两道题分别从无向边和有向边两种情况进行考虑。

二、题目与题解

题目一:108. 冗余连接

题目链接

108. 冗余连接 (kamacoder.com)

题目描述

有一个图,它是一棵树,他是拥有 n 个节点(节点编号1到n)和 n - 1 条边的连通无环无向图(其实就是一个线形图),如图:

现在在这棵树上的基础上,添加一条边(依然是n个节点,但有n条边),使这个图变成了有环图,如图:

先请你找出冗余边,删除后,使该图可以重新变成一棵树。

输入描述

第一行包含一个整数 N,表示图的节点个数和边的个数。

后续 N 行,每行包含两个整数 s 和 t,表示图中 s 和 t 之间有一条边。

输出描述

输出一条可以删除的边。如果有多个答案,请删除标准输入中最后出现的那条边。

输入示例

3
1 2
2 3
1 3

输出示例

1 3

提示信息

图中的 1 2,2 3,1 3 等三条边在删除后都能使原图变为一棵合法的树。但是 1 3 由于是标准输出里最后出现的那条边,所以输出结果为 1 3

数据范围

1 <= N <= 1000.

题解:并查集

这题就是昨天打卡模板的经典应用。

这里需要注意的是:

判断是否形成环:在将这对节点合并之前,会调用 isSame 函数来判断 s 和 t 是否已经在同一个集合中

如果 isSame(s, t) 返回 true,则说明 s 和 t 已经在同一个集合中,合并它们将会形成一个环。因此,程序会输出这对节点并结束

合并集合:如果 isSame 函数返回 false,说明 s 和 t 不在同一个集合中,合并它们不会形成环,则调用 join 函数将 s 和 t 所在的集合合并。

代码如下:

#include <bits/stdc++.h>
using namespace std;int n;
vector<int> father(101, 0);// 并查集初始化
void init()
{for (int i = 1; i <= n; i++) // 节点是从1到nfather[i] = i;           
}// 并查集里寻找该节点的根节点(带路径压缩)
int find(int u)
{if (u != father[u])father[u] = find(father[u]);return father[u];
}// 判断u和v是否找到同一个根节点 -- 是否在同一个集合中
bool isSame(int u, int v)
{int rootu = find(u);int rootv = find(v);return rootu == rootv;
}// join 函数用于合并两个节点所在的集合:将v->u这条边加入并查集
void join(int u, int v)
{int rootu = find(u);int rootv = find(v);if (rootu != rootv)father[rootv] = rootu;
}int main()
{cin >> n;init(); // 初始化并查集while (n--){int s, t;cin >> s >> t;if (isSame(s, t)){cout << s << " " << t;return 0;}elsejoin(s, t);}
}

题目二:109. 冗余连接II

题目链接

109. 冗余连接II (kamacoder.com)

题目描述

有一种有向树,该树只有一个根节点,所有其他节点都是该根节点的后继。该树除了根节点之外的每一个节点都有且只有一个父节点,而根节点没有父节点。有向树拥有 n 个节点和 n - 1 条边。如图: 

现在有一个有向图,有向图是在有向树中的两个没有直接链接的节点中间添加一条有向边。如图:

输入一个有向图,该图由一个有着 n 个节点(节点编号 从 1 到 n),n 条边,请返回一条可以删除的边,使得删除该条边之后该有向图可以被当作一颗有向树。

输入描述

第一行输入一个整数 N,表示有向图中节点和边的个数。 

后续 N 行,每行输入两个整数 s 和 t,代表这是 s 节点连接并指向 t 节点的单向边

输出描述

输出一条可以删除的边,若有多条边可以删除,请输出标准输入中最后出现的一条边。

输入示例

3
1 2
1 3
2 3

输出示例

2 3

提示信息

在删除 2 3 后有向图可以变为一棵合法的有向树,所以输出 2 3

数据范围:

1 <= N <= 1000.

题解:并查集

注意这题变成了有向图。

一个关键

在有向图中,如果一个节点的入度为2,那么它至少是环的一部分。这是因为至少存在两条边从不同的节点指向该节点,如果这些边之间没有其他的边连接,那么它们就构成了一个环 。

然后需要注意的是,对于有向图问题,我们需要用一个二维数组来存储所有有向边(该题中使用edges[][] 存储,edges[s][t]表示从s到t的有向边)

这里直接看代码:

#include <bits/stdc++.h>
using namespace std;int n;
vector<int> father(1001, 0);// 并查集初始化
void init()
{for (int i = 1; i <= n; i++) // 节点是从1到nfather[i] = i;
}// 并查集里寻找该节点的根节点(带路径压缩)
int find(int u)
{if (u != father[u])father[u] = find(father[u]);return father[u];
}// 判断u和v是否找到同一个根节点 -- 是否在同一个集合中
bool isSame(int u, int v)
{int rootu = find(u);int rootv = find(v);return rootu == rootv;
}// join 函数用于合并两个节点所在的集合:将v->u这条边加入并查集
void join(int u, int v)
{int rootu = find(u);int rootv = find(v);if (rootu != rootv)father[rootv] = rootu;
}// 在有向图里找到删除的那条边,使其变成树
void getRemoveEdge(const vector<vector<int>> &edges)
{init();                     // 初始化并查集for (int i = 0; i < n; i++) // 遍历所有的边{if (isSame(edges[i][0], edges[i][1])) // 如果构成有向环,则删除这条边{cout << edges[i][0] << " " << edges[i][1];return;}elsejoin(edges[i][0], edges[i][1]); // 否则将边加入并查集}
}// 删一条边之后判断是不是树
bool isTreeAfterRemoveEdge(const vector<vector<int>> &edges, int deleteEdge)
{init(); // 初始化并查集for (int i = 0; i < n; i++){if (i == deleteEdge)continue; // 跳过需要删除的边if (isSame(edges[i][0], edges[i][1]))return false; // 如果构成有向环,则不是树elsejoin(edges[i][0], edges[i][1]); // 将边加入并查集}return true; // 如果没有构成环,则是树
}int main()
{int s, t;vector<vector<int>> edges;      // 存储所有边 -- 有向:edge[s][t]表示边 s->tcin >> n;                       // 输入边的数量vector<int> inDegree(n + 1, 0); // 记录节点入度for (int i = 0; i < n; i++){cin >> s >> t;           // 输入边s->t(有向边)inDegree[t]++;           // 增加节点t的入度edges.push_back({s, t}); // 将边加入边列表 edge}vector<int> vec; // 存储入度为2的边的索引// 注意:找到入度为2的节点所对应的边,倒序遍历以优先删除最后出现的边for (int i = n - 1; i >= 0; i--){if (inDegree[edges[i][1]] == 2)vec.push_back(i); // 将边的索引加入vec}// 如果有入度为2的边,则尝试删除这些边if (vec.size() > 0){// 优先删除vec[0]这条边if (isTreeAfterRemoveEdge(edges, vec[0]))cout << edges[vec[0]][0] << " " << edges[vec[0]][1];elsecout << edges[vec[1]][0] << " " << edges[vec[1]][1];return 0;}// 如果没有入度为2的边,则一定有有向环,找到构成环的边并删除getRemoveEdge(edges);
}

三、小结

今天的打卡到此结束,图论的题很繁琐,需要花时间总结消化,后边会试着多练习这一块的内容。最后,希望终有所获。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/web/52541.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

互联网+教育中小学校园云解决方案

1. “互联网”教育的定义与目标 “互联网”教育是指利用信息技术&#xff0c;包括移动互联网、云计算、大数据和物联网等&#xff0c;推动教育变革和创新。其目标是构建一个网络化、数字化、个性化和终身化的教育体系&#xff0c;实现学习型社会的建设&#xff0c;培养创新人才…

【Linux:文件系统】

了解磁盘结构 盘片可读可写一面盘面有一个磁头&#xff0c;一个盘片俩磁头磁盘本质是一个机械设备磁盘中的盘片高速旋转是为了定为扇区磁盘中的磁头 左右摆动定义磁道磁盘的读写单位&#xff1a;4kb /512字节 如何找到一个指定的扇区 找到指定的磁头&#xff08;header)找到指定…

动手学深度学习(pytorch土堆)-03Transforms简单入门学习

1 torchvision中的transforms主要是对图片进行一些变换。 transforms结构及用法 将特定格式图片经过transforms里面的工具处理输出预期的图像 Totensor使用 tensor数据类型可以理解为包装了一些反向神经网络所需要的一些参数 PIL_Image读取 img_path"hymenoptera_da…

【软件测试】测试的分类

目录 &#x1f384;为什么要对软件测试进行分类&#xff1f; &#x1f333;按照测试目标分类 &#x1f6a9;界面测试 &#x1f6a9;功能测试 &#x1f6a9;性能测试 &#x1f6a9;可靠性测试 &#x1f6a9;安全性测试 &#x1f6a9;易用性测试 &#x1f3c0;标准性和规…

QT Creater实现国庆节主题项目【0基础完成版】

本文适用对象 想要学习qt creater的小白;想要学习c++制作软件的编程爱好者。可以先下载这篇博客绑定的资源,然后一边操作,一边学习,会更高效~0. 创建初始项目 一步步来操作吧,首先下载qt creter,之前发布过相关资源,大家直接查找下载,或者自行下载。 1. 初始代码 mai…

RLFD: Imitation Bootstrapped Reinforcement Learnin

CoRR 2023 paper 视觉或向量输入下的RLFD方法&#xff0c;利用模仿学习得到的策略&#xff0c;以自举方式引导在强化学习的策略优化以及价值函数优化。同时&#xff0c;为了更好的表征&#xff0c;IBRL选取VIT-based的网络以及添加Dropout 到policy。 Method 首先利用模仿学…

2024.9.11

在界面上显示当前时间&#xff0c;再设置一个闹钟&#xff0c;到了时间就吱吱响&#xff08;至少5遍&#xff09; #include "widget.h" #include "ui_widget.h"Widget::Widget(QWidget *parent): QWidget(parent), ui(new Ui::Widget),speecher(new QTe…

常见加解密算法09 - HASH 算法

各位读者你们好啊&#xff0c;今天讨论一下 HASH 算法&#xff0c;也是这个系列的完结篇&#xff01;&#xff01;&#xff01; Hash算法&#xff0c;又称散列算法&#xff0c;是一种从任意长度的数据字符串中创建小的、固定长度的值的函数&#xff0c;该值通常被视为数据的“指…

51单片机快速入门之点灯 STC 51单片机

第一步创建工程 第二步加载头文件 第三步编写代码 点灯完成 解释:主函数为main() 内部P1控制的是p1.0-p1.7 引脚 0为低电平

云计算实训41——部署project_exam_system项目(续)

# 创建脚本&#xff0c;可以在java环境中运行任何的jar包或者war包#!/bin/bash/usr/local/jdk/bin/java -jar /java/src/*.?ar一、思路分析 &#xff08;1&#xff09;nginx 1、下载镜像&#xff0c;将本地的dist项目的目录挂载在容器的/usr/share/nginx/html/ 2、启动容器 …

freertos 任务调度—抢占式, 时间片

FreeRTOS 操作系统支持三种调度方式&#xff1a; 抢占式调度&#xff0c;时间片调度和合作式调度。 实际应用主要是抢占式调度和时间片调度&#xff0c;合作式调度用到的很少. 1,抢占式调度 每个任务都有不同的优先级&#xff0c; 任务会一直运行直到被高优先级任务抢占或者遇到…

【MySQL】查询语句之inner、left、right、full join 的区别

前言&#xff1a; INNER JOIN 和 OUTER JOIN 是SQL中常用的两种连接方式&#xff0c;用于从两表活多表中提取相关的数据。两者区别主要在于返回的 结果集 如何处理 匹配 与 不匹配 的行。 目录 1、INNER JOIN 2、OUTER JOIN 3、总结 1、INNER JOIN 称为内连接&#xff0c;只…

SVGJS操作

svgjs用于操作 SVG 和动画的轻量级库。 官网 SVG.js v3.2 |家 (svgjs.dev) 效果 代码如下 <template><h3>测试操作已有SVG</h3><button click"changeText()">利用ID定位</button><button click"changeChild()">chan…

【Hot100】LeetCode—322. 零钱兑换

目录 1- 思路动态规划 2- 实现⭐322. 零钱兑换——题解思路 3- ACM 实现 原题链接&#xff1a;322. 零钱兑换 1- 思路 动态规划 动规五部曲 1- 定义 dp 数组确定含义 dp[j] 代表凑到金钱为 j 的最少硬币个数 2- 递推公式 dp[j] Math.min(dp[j],dp[amount-]1) 3- 初始化 dp[…

【苍穹外卖】前端 Day 1

1 Vue 1.1 通过 vue cli 脚手架创建前端工程 1.2 项目结构 1.3 启动项目 VS Code 启动前端项目&#xff1a; npm run serve 注意这里占用端口号 8080 与 java springboot 占用端口号一致&#xff0c;有冲突 serve 是这个名字 终止&#xff1a;ctrl c 修改端口号 2 vue 基本…

信刻光盘安全隔离与信息交换系统

随着各种数据传输、储存技术、信息技术的快速发展&#xff0c;保护信息安全是重中之重。军工、政府、部队及企事业单位等利用A网与B网开展相关工作已成为不可逆转的趋势。针对于业务需要与保密规范相关要求&#xff0c;涉及重要秘密信息&#xff0c;需做到安全的物理隔离&#…

离线版问卷-可集成到现有系统

目录标题 离线版问卷&#x1f4a1;前言亮点场景题外话 &#x1f3a8; 预览&#x1f308; 技术栈&#x1f4e6; 仓库&#x1f4bb; 初始化&#x1f680; 启动&#x1f6e0;️ 打包&#x1f5c2; 目录结构✨ 使用方法集成【设计问卷】集成【填写问卷】集成【只读问卷】集成【填答…

省委书记邀约大学生创业,长沙又一次为年轻人沸腾

敢想敢做的大学生&#xff0c;一直是创新创业的主力军。尤其是这些年“学术型”创业团队在各行各业越来越多见&#xff0c;市场对他们的接纳和支持力度也越来越强&#xff0c;给了新一代的大学生们更大的底气。 以往&#xff0c;大学生创业经常“落地生根”&#xff0c;先搞事…

【编译原理】编译器概述、编译器结构、编译器实例

编译器概述、编译器结构、编译器实例 编译器概述 1.编译器是一个程序 核心功能是把源代码翻译成目标代码 比如源代码&#xff1a;C/C&#xff0c;Java&#xff0c;C#&#xff0c;html 目标代码&#xff1a;X86&#xff0c;IA64,ARM,… 把一种源程序翻译成另外一种源程序&…

Facebook的秘密算法:如何提升你的社交体验

在数字时代&#xff0c;社交媒体平台已经成为我们日常生活的重要组成部分。作为全球最大的社交网络之一&#xff0c;Facebook通过其复杂的算法&#xff0c;影响着亿万用户的社交体验。这些算法不仅决定了我们在平台上看到的内容&#xff0c;还在背后默默优化我们的互动方式。本…