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

2024-01-20 21:52

本文主要是介绍Peter算法小课堂—拓扑排序与最小生成树,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

拓扑排序

讲拓扑排序前,我们要先了解什么是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

这篇关于Peter算法小课堂—拓扑排序与最小生成树的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/627463

相关文章

浅析如何使用Swagger生成带权限控制的API文档

《浅析如何使用Swagger生成带权限控制的API文档》当涉及到权限控制时,如何生成既安全又详细的API文档就成了一个关键问题,所以这篇文章小编就来和大家好好聊聊如何用Swagger来生成带有... 目录准备工作配置 Swagger权限控制给 API 加上权限注解查看文档注意事项在咱们的开发工作里,API

Spring排序机制之接口与注解的使用方法

《Spring排序机制之接口与注解的使用方法》本文介绍了Spring中多种排序机制,包括Ordered接口、PriorityOrdered接口、@Order注解和@Priority注解,提供了详细示例... 目录一、Spring 排序的需求场景二、Spring 中的排序机制1、Ordered 接口2、Pri

Java使用POI-TL和JFreeChart动态生成Word报告

《Java使用POI-TL和JFreeChart动态生成Word报告》本文介绍了使用POI-TL和JFreeChart生成包含动态数据和图表的Word报告的方法,并分享了实际开发中的踩坑经验,通过代码... 目录前言一、需求背景二、方案分析三、 POI-TL + JFreeChart 实现3.1 Maven

大数据小内存排序问题如何巧妙解决

《大数据小内存排序问题如何巧妙解决》文章介绍了大数据小内存排序的三种方法:数据库排序、分治法和位图法,数据库排序简单但速度慢,对设备要求高;分治法高效但实现复杂;位图法可读性差,但存储空间受限... 目录三种方法:方法概要数据库排序(http://www.chinasem.cn对数据库设备要求较高)分治法(常

MybatisGenerator文件生成不出对应文件的问题

《MybatisGenerator文件生成不出对应文件的问题》本文介绍了使用MybatisGenerator生成文件时遇到的问题及解决方法,主要步骤包括检查目标表是否存在、是否能连接到数据库、配置生成... 目录MyBATisGenerator 文件生成不出对应文件先在项目结构里引入“targetProje

Python使用qrcode库实现生成二维码的操作指南

《Python使用qrcode库实现生成二维码的操作指南》二维码是一种广泛使用的二维条码,因其高效的数据存储能力和易于扫描的特点,广泛应用于支付、身份验证、营销推广等领域,Pythonqrcode库是... 目录一、安装 python qrcode 库二、基本使用方法1. 生成简单二维码2. 生成带 Log

Python中的随机森林算法与实战

《Python中的随机森林算法与实战》本文详细介绍了随机森林算法,包括其原理、实现步骤、分类和回归案例,并讨论了其优点和缺点,通过面向对象编程实现了一个简单的随机森林模型,并应用于鸢尾花分类和波士顿房... 目录1、随机森林算法概述2、随机森林的原理3、实现步骤4、分类案例:使用随机森林预测鸢尾花品种4.1

Python中lambda排序的六种方法

《Python中lambda排序的六种方法》本文主要介绍了Python中使用lambda函数进行排序的六种方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们... 目录1.对单个变量进行排序2. 对多个变量进行排序3. 降序排列4. 单独降序1.对单个变量进行排序

Python使用Pandas库将Excel数据叠加生成新DataFrame的操作指南

《Python使用Pandas库将Excel数据叠加生成新DataFrame的操作指南》在日常数据处理工作中,我们经常需要将不同Excel文档中的数据整合到一个新的DataFrame中,以便进行进一步... 目录一、准备工作二、读取Excel文件三、数据叠加四、处理重复数据(可选)五、保存新DataFram

SpringBoot生成和操作PDF的代码详解

《SpringBoot生成和操作PDF的代码详解》本文主要介绍了在SpringBoot项目下,通过代码和操作步骤,详细的介绍了如何操作PDF,希望可以帮助到准备通过JAVA操作PDF的你,项目框架用的... 目录本文简介PDF文件简介代码实现PDF操作基于PDF模板生成,并下载完全基于代码生成,并保存合并P