Peter算法小课堂—拓扑排序与最小生成树

发布时间:2024年01月20日

拓扑排序

讲拓扑排序前,我们要先了解什么是DAG树。所谓DAG树,就是指“有向无环图”。请判断下列图是否是DAG图

第一幅图,它不是DAG图,因为它形成了一个环。第二幅图,它也不是DAG图,因为它没有方向。第三幅图才叫真正的DAG图(DAG图不一定联通)。

那什么叫DAG图的拓扑排序呢?排序大家都知道。拓扑排序指,按照一定次序(箭头方向)来遍历这幅图。

我们看道题吧。

太戈编程877题

题目描述:

你是一个电子游戏高手,正在研究一款新的游戏。该游戏共有n种关卡有待解锁,编号1到n。你发现关卡的解锁有m条依赖关系,第i条为:解锁关卡ai前必须先解锁关卡bi。请你为n个关卡设计一个可行的解锁顺序,若有多个可行解请输出字典序最小解。本题保证有解。

这道题中,我们怎样画这幅图呢?我们遵守一个原则“若u和v有一条有向边,说明u必须在v之前访问”。那具体怎么解决呢?下面我们介绍两种方法:Kahn算法和DFS实现。

Kahn算法

代码:

cin>>n>>m;
for(int i=1;i<=m;i++){
	int a,b;
	cin>>a>>b;
	if(d[b][a]) continue;//d[b][a]代表b要在a之前完成。易错点:重边处理
	d[b][a]=1;
	in[a]++;//in[a]代表关卡a的入度
}
for(int k=1,i;k<=n;k++){
	for(i=1;i<=n;i++) if(!vst[i]&&in[i]==0) break;
	topo[++cnt]=i;
	vst[i]=cnt;//vst[i]代表关卡i是否已解锁
	for(int j=1;j<=n;j++)
		if(d[i][j]) d[i][j]=0,in[j]--;
}
for(int i=1;i<=n;i++) cout<<topo[i]<<" ";

时空复杂度:O(N^2)

DFS

代码:

#include <bits/stdc++.h>
using namespace std;
const int N=10009;
vector<int> to[N];
int n,m,topo[N],cnt;
bool vst[N];
void dfs(int x){
	vst[x]=1;
	for(int i=0;i<to[N].size();i++){
		if(!vst[to[x][i]]) dfs(to[x][i]);
	}
	topo[n-cnt]=x;cnt++;
}
int main(){
	cin>>n>>m;
	for(int i=0;i<m;i++){
		int u,v;
		cin>>u>>v;
		to[u].push_back(v);
	}
	for(int i=1;i<=n;i++) if(!vst[i])dfs(i);
	for(int i=1;i<=n;i++) cout<<topo[i]<<" ";
	return 0;
}

我推荐使用DFS啊。练习:158、586。

后面来到我们的正(难)题:Minimum Spanning Tree 最小生成树

最小生成树

简称MST

在无向图中,任意两个顶点都有路径相通,称为连通图

连通图的生成树是包含原图n个顶和n-1条边的一棵树

最小生成树的所有边的长度综合是生成树里最小的

n个顶点的生成树有n-1条边,若再添加一条边,必定成环

大家算一下下列MST权值

答案:15 7。我们注意到第一幅图有6个点,也就是生成树有5条边,312564。第二幅图有4个点,生成树有3条边,1423。

下面,我们将要教大家如何解决最小生成树问题

Kruskal算法

贪心:每次找最短边,尝试加入最小生成树。

所以,大家要先会并查集,不会的小彭友看Peter算法小课堂—并查集-CSDN博客

给大家一个标程啊。

#include <bits/stdc++.h>
using namespace std;
const int N=109;
const int M=5009;
struct edge{int u,v,w;};
edge e[M];
int n,m,id[N];
int root(int u){
	return id[u]==u?u:id[u]=root(id[u]);
}
bool cmp(const edge&a,const edge&b){
	return a.w<b.w;
}
void Kruskal(){
	sort(e,e+m,cmp);
	for(int u=1;u<=n;u++) id[u]=u;
	int ans=0;
	for(int k=0;k<m;k++){
		int ru=root(e[k].u);
		int rv=root(e[k].v);
		if(ru==rv) continue;
		id[ru]=rv;
		ans+=e[k].w;
	}
	cout<<ans<<endl;
}
int main(){
	cin>>n>>m;
	for(int i=0;i<m;i++)
		cin>>e[i].u>>e[i].v>>e[i].w;
	Kruskal();
	return 0;
}

那有的人就会疑惑,为什么Kruskal算法能找到MST呢?下面给出证明。

请你证明:Kruskal选的第1条边e1一定在某棵MST中。

证:假设存在1棵不包含e1的MST记作T。向T中添加e1,必定成环。环中必有边长不小于e1的边f,删除f。新的生成树T+e1-f的边长总和不超过T,不符合最小条件。

可视化网址:最小生成树 MST (Prim算法,Kruskal算法) - VisuAlgo

文章来源:https://blog.csdn.net/zhang040818/article/details/135706383
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。