【深度优先搜索】【树】【有向图】【推荐】685. 冗余连接 II

news2025/1/17 3:55:41

LeetCode685. 冗余连接 II

在本问题中,有根树指满足以下条件的 有向 图。该树只有一个根节点,所有其他节点都是该根节点的后继。该树除了根节点之外的每一个节点都有且只有一个父节点,而根节点没有父节点。
输入一个有向图,该图由一个有着 n 个节点(节点值不重复,从 1 到 n)的树及一条附加的有向边构成。附加的边包含在 1 到 n 中的两个不同顶点间,这条附加的边不属于树中已存在的边。
结果图是一个以边组成的二维数组 edges 。 每个元素是一对 [ui, vi],用以表示 有向 图中连接顶点 ui 和顶点 vi 的边,其中 ui 是 vi 的一个父节点。
返回一条能删除的边,使得剩下的图是有 n 个节点的有根树。若有多个答案,返回最后出现在给定二维数组的答案。
示例 1:

输入:edges = [[1,2],[1,3],[2,3]]
输出:[2,3]
示例 2:
输入:edges = [[1,2],[2,3],[3,4],[4,1],[1,5]]
输出:[4,1]
提示:
n == edges.length
3 <= n <= 1000
edges[i].length == 2
1 <= ui, vi <= n

预备知识

有向图并集查找使用的2个前提:

前提一:所有的点出度为1或为0。
前提二:无环。
以当前节点为起点的最长路径终点,简称最长终点。如:两条边 A B → \overrightarrow{AB} AB B C → \overrightarrow{BC} BC ,A的最长终点是C。

有向树性质

性质一:任意节点都能访问自己所有后代,且不能访问自己后代之外的节点。
性质二:如果删除的边不是指向某节点的祖先,则根节点一定能访问此节点。

暴力解法

枚举删除的边,时间复杂度O(n);判断余下的边是否是有根树,时间复杂度O(n^n)。

深度优先判断有根树

前提三该树除了根节点之外的每一个节点都有且只有一个父节点,而根节点没有父节点。    ⟺    \iff 只有一个节点的入度为0,此节点为根。其它节点入度为1。
前提四该树只有一个根节点,所有其他节点都是该根节点的后继。    ⟺    \iff 根节点可以访问所有的节点。通过一条边直接访问的节点是根节点的后继。通过两条边访问的节点是后继的后继 ⋯ \cdots
如果有环一定不行,n-1条边,如果有重复节点则无法访问n-1个新节点。所以访问了重复的节点直接返回。

核心代码

class Solution {
public:
	vector<int> findRedundantDirectedConnection(vector<vector<int>>& edges) {
		m_c = edges.size();
		for (int i = m_c-1 ; i >=0 ;i--)
		{
			vector<vector<int>> vNeiBo(m_c);
			for (int j = 0; j < m_c; j++)
			{
				if (j != i)
				{
					vNeiBo[edges[j][0] - 1].emplace_back(edges[j][1] - 1);
				}
			}
			if (Is(vNeiBo))
			{				
				return edges[i];
			}
		}
		return {};
	}
	bool Is(const vector<vector<int>>& vNeiBo)const
	{
		vector<int> vInDeg(m_c);
		for (const auto& v : vNeiBo)
		{
			for (const auto& to : v)
			{
				vInDeg[to]++;
			}			
		}
		for (int i = 0; i < m_c; i++)
		{
			if (0 == vInDeg[i])
			{
				vector<int> vVis(m_c);
				DFS(vVis, vNeiBo, i);
				return m_c == std::accumulate(vVis.begin(), vVis.end(), 0);
			}
		}
		return false;
	}
	void DFS(vector<int>& vVis, const vector<vector<int>>& vNeiBo,int cur)const
	{
		if (vVis[cur])
		{
			return;
		}
		vVis[cur] = 1;
		for (const auto& next : vNeiBo[cur])
		{
			DFS(vVis, vNeiBo, next);
		}
	}
	int m_c;

};

测试用例

template<class T,class T2>
void Assert(const T& t1, const T2& t2)
{
	assert(t1 == t2);
}

template<class T>
void Assert(const vector<T>& v1, const vector<T>& v2)
{
	if (v1.size() != v2.size())
	{
		assert(false);
		return;
	}
	for (int i = 0; i < v1.size(); i++)
	{
		Assert(v1[i], v2[i]);
	}

}

int main()
{
	vector<vector<int>> edges;
	{
		Solution sln;
		edges = { {5,2},{5,1},{3,1},{3,4},{3,5} };
		auto res = sln.findRedundantDirectedConnection(edges);
		Assert({ 3,1 }, res);
	}
	{
		Solution sln;
		edges = { {1, 2}, { 2,3 }, { 3,1 }, { 4,1 } };
		auto res = sln.findRedundantDirectedConnection(edges);
		Assert({ 3,1 }, res);
	}
	{
		Solution sln;
		edges = { {1,2},{1,3},{2,3} };
		auto res = sln.findRedundantDirectedConnection(edges);
		Assert({ 2,3 }, res);
	}
	{
		Solution sln;
		edges = { {1,2},{2,3},{3,4},{4,1},{1,5} };
		auto res = sln.findRedundantDirectedConnection(edges);
		Assert({ 4,1 }, res);
	}
	{
		Solution sln;
		edges = { {2,1},{3,1},{4,2},{1,4} };
		auto res = sln.findRedundantDirectedConnection(edges);
		Assert({ 2,1 }, res);
	}
	{
		Solution sln;
		edges = { {4,2},{1,5},{5,2},{5,3},{2,4} }		;
		auto res = sln.findRedundantDirectedConnection(edges);
		Assert({ 4,2 }, res);
	}
}

并集查找

所有边反向,即:根节出度为0,其它节点出度为1。所有节点的最长终点是都是根节点。即判定一:所有节点的最终节点相同。
有向树是判断一,无需证明。下面来证明判断一一定是有向树,判定一符合前提四无需证明,只需证明判定一符合前提三。
{ n − 1 个非根节点最终目的地是根节点,说明这些定点的出度至少为 1 。 总共 n − 1 条边。 → { 根节点出度为 0 其它节点出度为 1 \begin{cases} n-1个非根节点最终目的地是根节点,说明这些定点的出度至少为1。\\ 总共n-1条边。\\ \end{cases}\rightarrow \begin{cases}根节点出度为0\\ 其它节点出度为1 \end{cases} {n1个非根节点最终目的地是根节点,说明这些定点的出度至少为1总共n1条边。{根节点出度为0其它节点出度为1

简化

n-1条边,如果无环,则说明访问了n-1个新节点。且只有一个连通区域,否则顶点更多。故:
无环没有出度为2的定点    ⟺    \iff 有向树

代码

class CUnionFindDirect
{
public:
	CUnionFindDirect(int iSize)
	{
		m_vRoot.resize(iSize);
		iota(m_vRoot.begin(), m_vRoot.end(), 0);
	}
	void Connect(bool& bConflic, bool& bCyc, int iFrom, int iTo)
	{
		bConflic = bCyc = false;
		if (iFrom != m_vRoot[iFrom])
		{
			bConflic = true;
		}

		Fresh(iTo);
		if (m_vRoot[iTo] == iFrom)
		{
			bCyc = true;
		}
		if (bConflic || bCyc)
		{
			return;
		}

		m_vRoot[iFrom] = m_vRoot[iTo];
	}
private:
	int Fresh(int iNode)
	{
		if (m_vRoot[iNode] == iNode)
		{
			return iNode;
		}
		return m_vRoot[iNode] = Fresh(m_vRoot[iNode]);
	}
	vector<int> m_vRoot;
};

class Solution {
public:
	vector<int> findRedundantDirectedConnection(vector<vector<int>>& edges) {
		m_c = edges.size();
		for (int i = m_c-1 ; i >=0 ;i--)
		{			
			if (Is(edges,i))
			{				
				return edges[i];
			}
		}
		return {};
	}
	bool Is(const vector<vector<int>>& edges,int i )const
	{
		CUnionFindDirect uf(m_c);
		for (int j = 0; j < m_c; j++)
		{
			if (j != i)
			{
				bool b2 = false, bCyc = false;
				uf.Connect(b2,bCyc,edges[j][1] - 1, edges[j][0] - 1);
				if (b2 || bCyc)
				{
					return false;
				}
			}
		}
		return true;
	}

	int m_c;

};

特殊情况

如果有环,计算最长终点,可能会死循环。所以要提前判断。有环    ⟺    \iff B的最长终点是A,发现新边 A B → \overrightarrow{AB} AB
必要条件无需证明环是: B → A → B 。 必要条件无需证明环是:B \rightarrow A \rightarrow B 。 必要条件无需证明环是:BAB
现在证明充分条件:
由于是环,所以B一定能到A,假定增加新边 A B → \overrightarrow{AB} AB 前,B的最长终点是C,那么A能访问C,也就是A出度至少为1。加上 A B → \overrightarrow{AB} AB 出度就成了2。和前提三矛盾。

优化

有向森林B的父节点是C,增加的边 A B → \overrightarrow{AB} AB 。有以下两种情况:
一,无环。如果B不是A的祖先,则B无法访问A,增加新边后,就无法形成环。存在入度为2的顶点
1.1,和森林的某边重合。删除任意一边。
在这里插入图片描述

1.2,前向边,A是祖先,B是后代(A能访问B)。
在这里插入图片描述
如果先发现AB,后发现CB则是横叉边。
1.3,横叉变。A和B都无法到达他们的公共祖先,所以不会有环。
在这里插入图片描述
如果先发现AB,后发现CB则是前向边。
总结: 根据性质二,B不会是A祖先,所以1.2和1.3删除CB不会影响根节点访问A。

二,后向边,B是祖先,A是后代,B能访问A。增加新边后,A能访问B。形成了环。
2.1,如果B是根节点,则这个环上所有节点都能访问B,也就是能访问所有节点。删除任意一边后,会形成新树。不存在入度为2的顶点
在这里插入图片描述

,2.2 如果B不是根节点。不包括 C B → \overrightarrow{CB} CB A B → \overrightarrow{AB} AB 如果C是根节点的后代,则删除 A B → \overrightarrow{AB} AB 可以形成新树 。如果A是根节点的后代,则删除 C B → \overrightarrow{CB} CB ,可形成新树 。存在入度为2的顶点

在这里插入图片描述

无论是发现环,还是发现入度为2,还是同时发现环和入度为2,都只能删除折线。
处理方法一: { 删除 C B → 不通过 C B → ,根节点能到达 A 。 删除 A B → 不通过 A B → ,根节点能到达 C 。 处理方法一:\begin{cases} 删除\overrightarrow{CB} & 不通过\overrightarrow{CB},根节点能到达A。\\ 删除\overrightarrow{AB} &不通过\overrightarrow{AB},根节点能到达C。\\ \end{cases} 处理方法一:{删除CB 删除AB 不通过CB ,根节点能到达A不通过AB ,根节点能到达C

总结,5种情况合并两种:
{ 删除环中任意一边(最后一边) 不存在入度为 2 的定点 处理方法一 o t h e r \begin{cases} 删除环中任意一边(最后一边) & 不存在入度为2的定点 \\ 处理方法一 & other \\ \end{cases} {删除环中任意一边(最后一边)处理方法一不存在入度为2的定点other

代码

class CUnionFindDirect
{
public:
	CUnionFindDirect(int iSize)
	{
		m_vRoot.resize(iSize);
		iota(m_vRoot.begin(), m_vRoot.end(), 0);
	}
	void Connect(bool& bConflic, bool& bCyc, int iFrom, int iTo)
	{
		bConflic = bCyc = false;
		if (iFrom != m_vRoot[iFrom])
		{
			bConflic = true;
		}

		Fresh(iTo);
		if (m_vRoot[iTo] == iFrom)
		{
			bCyc = true;
		}
		if (bConflic || bCyc)
		{
			return;
		}

		m_vRoot[iFrom] = m_vRoot[iTo];
	}
	int GetMaxDest(int iFrom)
	{
		Fresh(iFrom);
		return m_vRoot[iFrom];
	}	
private:
	int Fresh(int iNode)
	{
		if (m_vRoot[iNode] == iNode)
		{
			return iNode;
		}
		return m_vRoot[iNode] = Fresh(m_vRoot[iNode]);
	}
	vector<int> m_vRoot;
};

class Solution {
public:
	vector<int> findRedundantDirectedConnection(vector<vector<int>>& edges) {
		m_c = edges.size();
		vector<int> vParent(m_c, -1);
		int iParentOld = -1;
		for (int j = 0; j < m_c; j++)
		{
			if (vParent[edges[j][1] - 1] >= 0)
			{
				iParentOld = vParent[edges[j][1] - 1];
			}
			vParent[edges[j][1] - 1] = edges[j][0] - 1;
		}
		const int root = std::find(vParent.begin(), vParent.end(), -1) - vParent.begin();
		CUnionFindDirect uf(m_c);		
		int iCyc = -1;
		int i2 = -1;
		for (int j = 0; j < m_c; j++)
		{
			bool b2 = false, bCyc = false;
			uf.Connect(b2, bCyc, edges[j][1] - 1, edges[j][0] - 1);
			if (b2)
			{
				i2 = j;
			}
			if (bCyc)
			{
				iCyc = j ;
			}
		}
		if (-1 != i2)
		{
			const int dest = uf.GetMaxDest(edges[i2][1] - 1);
			return (root == dest) ? edges[i2] : vector<int>{ iParentOld + 1, edges[i2][1] };
		}

		return edges[iCyc];
	}
	int m_c;
};

注意

一,所有边都加到并集查找才判断最长终点。
二,由于冲突边和环边会忽略。所以可能出现 忽略环边时,让冲突边也消失。碰巧可以按环处理。
在这里插入图片描述
BA和CD已经发现,CB未发现。发现AB是环边,忽略。由于AB被忽略,所以CB不是冲突边。

娱乐性优化

情况2.2进一步分析,无论何种发现顺序都是删除环冲突边:
2.2.1,同时发现环边和冲突边。环冲突边最后发现。删除冲突边。
2.2.2 先发现环边,环边是冲突边。无法发先冲突边。删除环边。
2.2.3先发现环边,环边不是冲突边。能发现冲突边。删除另外一条冲突边。
2.2.4 先发现冲突边,忽略冲突环边,无法发现环。删除冲突边。
2.2.5 先发现冲突边,忽略非环冲突边,能发现环。删除另外一条冲突边。
总结: { 删除冲突边 发现冲突边未发现环边 1 , 2.2.4 删除环边 发现环边未发现冲突边 2.1 , 2.22 删除冲突边(环边 ) 冲突边就是环边 2.2.1 删除另外一条冲突边 发现冲突边和环边两者不等 2.2.3 , 2.25 总结:\begin{cases} 删除冲突边 & 发现冲突边未发现环边 & 1,2.2.4 \\ 删除环边 & 发现环边未发现冲突边 &2.1, 2.22 \\ 删除冲突边(环边) & 冲突边就是环边 & 2.2.1 \\ 删除另外一条冲突边 & 发现冲突边和环边两者不等 & 2.2.3,2.25\\ \end{cases} 总结: 删除冲突边删除环边删除冲突边(环边)删除另外一条冲突边发现冲突边未发现环边发现环边未发现冲突边冲突边就是环边发现冲突边和环边两者不等1,2.2.42.1,2.222.2.12.2.3,2.25

扩展阅读

视频课程

有效学习:明确的目标 及时的反馈 拉伸区(难度合适),可以先学简单的课程,请移步CSDN学院,听白银讲师(也就是鄙人)的讲解。
https://edu.csdn.net/course/detail/38771

如何你想快

速形成战斗了,为老板分忧,请学习C#入职培训、C++入职培训等课程
https://edu.csdn.net/lecturer/6176

相关下载

想高屋建瓴的学习算法,请下载《喜缺全书算法册》doc版
https://download.csdn.net/download/he_zhidan/88348653

我想对大家说的话
闻缺陷则喜是一个美好的愿望,早发现问题,早修改问题,给老板节约钱。
子墨子言之:事无终始,无务多业。也就是我们常说的专业的人做专业的事。
如果程序是一条龙,那算法就是他的是睛

测试环境

操作系统:win7 开发环境: VS2019 C++17
或者 操作系统:win10 开发环境: VS2022 C++17
如无特殊说明,本算法用**C++**实现。

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

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

相关文章

Python学习(16)|列表_遍历_排序_max_min_sum

列表的遍历&#xff1a; a [10,20,30,40] for obj in a: #obj 是临时变量名称&#xff0c;随意起名print(obj) 执行结果&#xff1a; 复制列表所有的元素到新列表对象&#xff1a; list1 [30,40,50] list2 list1 #只是将list2也指向了列表对象。也就是说list…

基于Python3的数据结构与算法 - 01 复杂度和列表查找

一、时间复杂度 定义&#xff1a;用来评估算法运行效率的一个式子。 例如&#xff1a;此处的O(1) 详单与一个时间单位 接下来我们看下面两个式子&#xff1a; 如果按照上面的定义&#xff0c;那么打印三次相当O(3)&#xff0c;下面的循环相当于O(n21) 但是实际不是这样的 因…

YOLO v5项目实战 P5 解决运行detect文件时设置了--view-img但是显示不出来的问题

up主讲的实时显示目标检测后的图片的两种方法&#xff1a; &#xff08;1&#xff09;在下面的Terminal中输入下列命令&#xff1a; python detect.py --view-img &#xff08;2&#xff09;点击进入右上方的detect的Edit Configurations 然后在这个参数这里输入 --view img…

Pulsar-架构与设计

Pulsar架构与设计 一、背景和起源二、框架概述1.设计特点2.框架适用场景 三、架构图1.Broker2.持久化存储&#xff08;Persistent storage&#xff09;3.Pulsar元数据&#xff08;Metadata store&#xff09; 四、功能特性1.消息顺序性2.消息回溯3.消息去重4.消息重投递5.消息重…

一键安装ROS适用于Ubuntu22/20/18

一键安装ROS适用于Ubuntu22/20/18 1、简介 ROS&#xff08;Robot Operating System&#xff0c;机器人操作系统&#xff09;是一个用于机器人软件开发的框架。它提供了一套工具和库&#xff0c;用于机器人应用程序的开发、测试和部署。ROS是由美国斯坦福大学机器人实验室&…

银河麒麟操作系统自动同步时间更新

1、银河麒麟操作系统基于Centos8的&#xff0c;因centos8取消了ntp服务器&#xff0c;所以导致之前使用ntpdate命令无法同步时间 2、centos默认使用chrony模块来进行同步时间 3、修改chrony配置同步时间服务器 vim /etc/chrony.conf 4、目前使用的是阿里云的时间服务器&…

2024年初中生古诗文大会安排预测和备考建议、资料

今天&#xff08;2024年2月19日&#xff09;是上海市中小学正式开学的日子&#xff0c;神兽归位&#xff0c;各位家长可以松口气了&#xff0c;为孩子规划今年的竞赛、学科活动了。根据一些家长朋友的咨询&#xff0c;今天六分成长介绍2024年中学生古诗文大会&#xff08;初中组…

图数据库 之 Neo4j - Cypher语法基础(5)

节点(Nodes) Cypher使用()来表示一个节点。 () # 最简单的节点形式&#xff0c;表示一个任意无特征的节点&#xff0c;其实就是一个空节点(movie) # 如果想指向一个节点在其他地方&#xff0c;我们可以给节点添加一个变量名(如movie)&#xff0c;表示一个变量名为 movie的节点…

SQL Developer 小贴士:Unshared Worksheet

在Oracle SQL Developer中&#xff0c;最常用的功能应该是SQL Worksheet&#xff0c;或Worksheet。 可以创建两类Worksheet&#xff0c;即Worksheet和Unshared Worksheets。前者是共享数据库连接的&#xff0c;后者会单独创建自己的连接。前者的快捷键是AltF10&#xff1b;后者…

SSM项目启动错误[main] ERROR org.springframework.boot.SpringApplication

[main] ERROR org.springframework.boot.SpringApplication - Application run failed org.yaml.snakeyaml.error.YAMLException: java.nio.charset.MalformedInputException: Input length 1 这个错误是由于Spring Boot应用程序在解析YAML文件时遇到了字符编码问题。java.ni…

国产降压芯片 小封装 外围简单 支持36V 48V 60V 85V 100V电压-H6225L

国产降压芯片&#xff0c;也就是由中国企业设计和生产的用于电压转换的集成电路芯片&#xff0c;随着国内半导体产业的快速发展&#xff0c;这些芯片已经开始在一些领域显示出其优势。主要优势涵盖以下几个方面&#xff1a; 成本优势&#xff1a;国产降压芯片因为本土化生产&a…

NLP_GPT生成式自回归模型

文章目录 介绍完整代码小结 介绍 自回归(Autoregressive)是自然语言处理模型的一种训练方法&#xff0c;其核心思想是基于已有的序列(词或字符)来预测下一个元素。在GPT中&#xff0c;这意味着模型会根据给定的上文来生成下一个词&#xff0c;如图所示。 在GPT模型的训练和推…

数据分析 — 动画图 pyecharts

目录 一、概念二、安装和导入三、绘图逻辑四、绘图1、柱状图2、折线图3、散点图4、饼图5、南丁格尔图6、Geo() 地理坐标第7、Map() 绘制区域8、词云图9、层叠图10、3D 图11、仪表板 一、概念 Pyecharts 是一个基于 Echarts 的 Python 可视化库&#xff0c;它通过 Python 生成 …

猪圈Pigsty-PG私有RDS集群搭建教程

博客 https://songxwn.com/Pigsty-PG-RDS/ 简介 Pigsty 是一个更好的本地自建且开源 RDS for PostgreSQL 替代&#xff0c;具有以下特点&#xff1a; 开箱即用的 PostgreSQL 发行版&#xff0c;深度整合地理、时序、分布式、图、向量、分词、AI等 150 余个扩展插件&#xff…

OpenAI Sora —— 文生视频为何如此逼真?AI算法架构解析

OpenAI于2024年2月16日发布了名为Sora的文生视频模型。Sora是一个革命性的视频生成模型&#xff0c;可以根据用户输入的简单文本脚本自动生成与好莱坞级别画面相媲美的视频内容&#xff0c;其生成的视频不仅仅是对已有素材的拼接或剪辑合成&#xff0c;而是从像素级别全新“绘制…

【Jvm】性能调优(上)线上问题排查工具汇总

文章目录 一.互联网概念1.产品闭环和业务闭环2.软件设计中的上游和下游3.JDK运行时常量池 二.CPU相关概念1.查询CPU信息2.CPU利用率&#xff08;CPU utilization&#xff09;和 CPU负载&#xff08;CPU load&#xff09;2.1.如何理解CPU负载2.2.top命令查看CPU负载均值2.3.CPU负…

动态头部:统一目标检测头部与注意力

论文地址:https://arxiv.org/pdf/2106.08322.pdf ai阅读论文_论文速读_论文阅读软件-网易有道速读 创新点是什么? 这篇文档的创新点是提出了一种统一的方法&#xff0c;将对象检测头和注意力机制结合起来。作者在文中提出了一种称为Dynamic Head的方法&#xff0c;通过引入…

Android 15 第一个开发者预览版

点击查看&#xff1a;first-developer-preview-android15 点击查看&#xff1a;Get Android 15 2024年2月16日,谷歌发布 Android 15 第一个开发者预览版 翻译 由工程副总裁戴夫伯克发布 今天&#xff0c;我们发布了Android 15的首个开发者预览版&#xff0c;这样我们的开发者就…

[技术杂谈]Chat With RTX 介绍

英伟达&#xff08;Nvidia&#xff09;已于近日发布了名为“Chat with RTX”的Demo版个性化AI聊天机器人&#xff0c;并在其海外官网渠道中提供了下载链接。 据了解&#xff0c;这是一款适用于Windows平台的聊天机器人&#xff0c;由TensorRT-LLM提供支持&#xff0c;完全在本地…

机器人革命:从斯坦福的通用操作接口到OpenAI的Sora,塑造未来的合成学习

引言 在机器人成为平凡工匠和前沿先驱的时代&#xff0c;我们正站在新黎明的边缘。本文将探讨斯坦福大学的通用操作接口&#xff08;UMI&#xff09;及其与OpenAI的Sora如何共同推进机器人技术&#xff0c;开创未来学习的新纪元。 正文 斯坦福的通用操作接口&#xff08;UMI…