Kruskal最小生成树【详细解释+动图图解】【sort中的cmp函数】 【例题:洛谷P3366 【模板】最小生成树】

本文主要是介绍Kruskal最小生成树【详细解释+动图图解】【sort中的cmp函数】 【例题:洛谷P3366 【模板】最小生成树】,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

  • Kruskal算法简介
    • Kruskal算法前置知识
      • sort 中的cmp函数
  • 算法思考
    • 样例详细示范与解释
    • kruskal模版code↓
  • 例题:洛谷P3366 【模板】最小生成树code↓
  • 完结撒花QWQ

Kruskal算法简介

K r u s k a l Kruskal Kruskal 是基于贪心算法 M S T MST MST 算法,核心思想为以边为中心查找最小生成树,时间复杂度 O ( m l o g 2 m ) O(mlog_{2}m) O(mlog2m),其中的 m m m 为边数

具体算法可分为两个步骤

1.以边权为优先级来进行排序

2.使用并查集查找连通性,如果不连通,则加边,加答案


Kruskal算法前置知识

1.对于 v e c t o r vector vector 的容器排序算法(使用 s o r t sort sort 即可)

sort(T.begin(),T.end(),cmp);//这是vector的排序方法

解释: T . b e g i n ( ) T.begin() T.begin() v e c t o r vector vector 的起始部分, T . e n d ( ) T.end() T.end() v e c t o r vector vector 的结束部分, T T T v e c t o r vector vector 的容器名

sort 中的cmp函数

c m p cmp cmp s o r t sort sort 重构函数,需要自己定义,这个函数的类型 b o o l bool bool内部变量的类型便是需要排序的容器的类型

cmp模版code如下↓

T name;
bool cmp(T x,T y){return x op y}
sort(name(first),name(last),cmp)

T T T容器类型 n a m e name name容器名字 n a m e ( f i r s t ) name(first) name(first) 代表容器的第一位 n a m e ( l a s t ) name(last) name(last) 表示容器的最后一位


2.使用结构体的构造来赋值

Edge(int a,int b,int c):u(a),v(b),w(c){};

上述构造函数的代码的意思等同于↓:

Edge(int a,int b,int c){u=a,v=b,w=c;}

在结构体里加边的操作也就为:T.push_back(Edge(u,v,w));


3.容器 v e c t o r vector vector 的定义

我们需要用容器来管理结构体

也就是将结构体给定义在容器里

vector<Node> T;//其中T为容器名,Node为结构体名

定义code总结↓:

struct Node{int u,v,w;//定义类型Edge(int a,int b,int c):u(a),v(b),w(c){};//使用构造
};
bool (Node x,Node y){return x.w<y.w}//具体使用vector里的哪一个定义排序的函数
vector<Node> T;//使用容器来管理结构体
sort(T.begin(),T.end(),cmp)//其中T为容器名

算法思考

我们先给出一个题目来进行思考↓:

x x x 市共有 n n n 个岛屿, m m m 种修桥的方案由于 x x x 市的市长是一个黑心市长,所以他想要选择一种方案使得总共修桥的钱最少
每年他可以修一座桥,问:需要几年才能使得所有的岛屿之间都可以互相同行,最少修桥的钱为多少?

我们可以知道:修桥的钱数就是边权,岛屿的名字就是点的编号

第一个问题很好解答,使得所有点之间都可以连通的最少边数 N − 1 N-1 N1 条边

第二个问题我们就需要进行 K r u s k a l Kruskal Kruskal 进行求最小生成树

输入格式为
1 1 1 行,两个整数 n n n , m m m
2 2 2 ~ n + 1 n+1 n+1 行,每行三个整数 u u u , v v v , w w w ,表示所连接的两点及其边权

我们先给出一组样例↓

4 6
1 2 11
2 3 13
3 4 9
4 1 21
1 3 23
4 2 20

样例解释如图示↓
在这里插入图片描述

样例详细示范与解释

因为我们是需要" 花最少的钱,办最多的事 ",所以我们需要先以边的权值为优先级进行排序,结果为↓

3 4 9
1 2 11
2 3 13
4 2 20
4 1 21
1 3 23

那么我们就可以开始进行判断了,每一次重复的过程为:查找两个点是否连通,如果不连通,则加边

int x=find(wei[i].u),y=find(wei[i].v);//查找两个点的祖先if(x!=y){//如果祖先相同,则他们连通,在同一个集合内f[x]=y;//将两条边连在一起ans+=wei[i].w;//将它的权值加在最终答案里cnt++;//已经连接的边数+1}

解释:因为我们最开始已经排过序了,所以如果不连通,那么这条边一定是连接这两个点的最小代价

最后,如果两个点不连通,直接加边和答案,如果边数已经满足最少边数 N − 1 N-1 N1 条,则返回答案

return cnt==n-1?ans:-1;//如果边数是n-1条则返回答案,否则没有答案,无法连接所有边

如何使用并查集查找两个点的连通性,可见我的另一篇博文:并查集【模版】& 路径压缩优化

动图视频如下:

kruskal模版code↓

int kruskal(int n,int m,vector<Edge> &wei){sort(wei.begin(),wei.end(),cmp);int ans=0,cnt=0;for(int i=0;i<m;i++){int x=find(wei[i].u),y=find(wei[i].v);if(x!=y){f[x]=y;ans+=wei[i].w;cnt++;}}return cnt==n-1?ans:-1;
}

例题:洛谷P3366 【模板】最小生成树code↓

#include <bits/stdc++.h>
using namespace std;
const int maxn=2e5+7;
struct Edge{int u,v,w;Edge(int a,int b,int c):u(a),v(b),w(c){};
};
int f[maxn]={},n,m;
bool cmp(Edge x,Edge y){return x.w<y.w;
}
int find(int x){return f[x]==x?x:f[x]=find(f[x]);} 
vector<Edge> wei;
int kruskal(int n,int m,vector<Edge> &wei){sort(wei.begin(),wei.end(),cmp);int ans=0,cnt=0;for(int i=0;i<m;i++){int x=find(wei[i].u),y=find(wei[i].v);if(x!=y){f[x]=y;ans+=wei[i].w;cnt++;}}return cnt==n-1?ans:-1;
}
int init(){for(int i=1;i<=n;i++) f[i]=i;return 0;
}
int main(){cin>>n>>m;for(int i=1;i<=n;i++) f[i]=i;for(int i=1;i<=m;i++){int u,v,w;cin>>u>>v>>w;wei.push_back(Edge(u,v,w));//因为是无向图,所以需要反过来再加一次边wei.push_back(Edge(v,u,w));}int ans=kruskal(n,2*m,wei);//因为是无向图,所以边数是原边数的两倍if(ans==-1) cout<<"orz";else cout<<ans;return 0;
}

这么一点代码当然是可以 A C AC AC

在这里插入图片描述

完结撒花QWQ

这篇关于Kruskal最小生成树【详细解释+动图图解】【sort中的cmp函数】 【例题:洛谷P3366 【模板】最小生成树】的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

如何在 Spring Boot 中实现 FreeMarker 模板

《如何在SpringBoot中实现FreeMarker模板》FreeMarker是一种功能强大、轻量级的模板引擎,用于在Java应用中生成动态文本输出(如HTML、XML、邮件内容等),本文... 目录什么是 FreeMarker 模板?在 Spring Boot 中实现 FreeMarker 模板1. 环

IntelliJ IDEA 中配置 Spring MVC 环境的详细步骤及问题解决

《IntelliJIDEA中配置SpringMVC环境的详细步骤及问题解决》:本文主要介绍IntelliJIDEA中配置SpringMVC环境的详细步骤及问题解决,本文分步骤结合实例给大... 目录步骤 1:创建 Maven Web 项目步骤 2:添加 Spring MVC 依赖1、保存后执行2、将新的依赖

Python的time模块一些常用功能(各种与时间相关的函数)

《Python的time模块一些常用功能(各种与时间相关的函数)》Python的time模块提供了各种与时间相关的函数,包括获取当前时间、处理时间间隔、执行时间测量等,:本文主要介绍Python的... 目录1. 获取当前时间2. 时间格式化3. 延时执行4. 时间戳运算5. 计算代码执行时间6. 转换为指

如何为Yarn配置国内源的详细教程

《如何为Yarn配置国内源的详细教程》在使用Yarn进行项目开发时,由于网络原因,直接使用官方源可能会导致下载速度慢或连接失败,配置国内源可以显著提高包的下载速度和稳定性,本文将详细介绍如何为Yarn... 目录一、查询当前使用的镜像源二、设置国内源1. 设置为淘宝镜像源2. 设置为其他国内源三、还原为官方

最详细安装 PostgreSQL方法及常见问题解决

《最详细安装PostgreSQL方法及常见问题解决》:本文主要介绍最详细安装PostgreSQL方法及常见问题解决,介绍了在Windows系统上安装PostgreSQL及Linux系统上安装Po... 目录一、在 Windows 系统上安装 PostgreSQL1. 下载 PostgreSQL 安装包2.

Python正则表达式语法及re模块中的常用函数详解

《Python正则表达式语法及re模块中的常用函数详解》这篇文章主要给大家介绍了关于Python正则表达式语法及re模块中常用函数的相关资料,正则表达式是一种强大的字符串处理工具,可以用于匹配、切分、... 目录概念、作用和步骤语法re模块中的常用函数总结 概念、作用和步骤概念: 本身也是一个字符串,其中

MySql match against工具详细用法

《MySqlmatchagainst工具详细用法》在MySQL中,MATCH……AGAINST是全文索引(Full-Textindex)的查询语法,它允许你对文本进行高效的全文搜素,支持自然语言搜... 目录一、全文索引的基本概念二、创建全文索引三、自然语言搜索四、布尔搜索五、相关性排序六、全文索引的限制七

python中各种常见文件的读写操作与类型转换详细指南

《python中各种常见文件的读写操作与类型转换详细指南》这篇文章主要为大家详细介绍了python中各种常见文件(txt,xls,csv,sql,二进制文件)的读写操作与类型转换,感兴趣的小伙伴可以跟... 目录1.文件txt读写标准用法1.1写入文件1.2读取文件2. 二进制文件读取3. 大文件读取3.1

Linux内核参数配置与验证详细指南

《Linux内核参数配置与验证详细指南》在Linux系统运维和性能优化中,内核参数(sysctl)的配置至关重要,本文主要来聊聊如何配置与验证这些Linux内核参数,希望对大家有一定的帮助... 目录1. 引言2. 内核参数的作用3. 如何设置内核参数3.1 临时设置(重启失效)3.2 永久设置(重启仍生效

IDEA自动生成注释模板的配置教程

《IDEA自动生成注释模板的配置教程》本文介绍了如何在IntelliJIDEA中配置类和方法的注释模板,包括自动生成项目名称、包名、日期和时间等内容,以及如何定制参数和返回值的注释格式,需要的朋友可以... 目录项目场景配置方法类注释模板定义类开头的注释步骤类注释效果方法注释模板定义方法开头的注释步骤方法注