# 最小生成树(Minimum Spanning Tree)
经典的最小生成树算法的主要提出者是kruskal和prim,所以今天我们介绍的也是kruskal算法和prim算法。简单来说最小生成树就是用最少的代价使得一个图连通。
下面来拿个图来举例子
kruskal算法:
1:先对每条的权重按从小到大排序;
2:每次选取其中没有被选过的最小权重的边,并且不能形成环
3:一直搜索到边数=点数-1,如果不能则该图不连通
给出图示,便于理解
最后算出他的总代价sum=1+2+4+5+10=22 附上模板代码,大家可以在洛谷上进行测试https://www.luogu.org/problem/P3366
题目描述
如题,给出一个无向图,求出最小生成树,如果该图不连通,则输出orz
输入格式
第一行包含两个整数N、M,表示该图共有N个结点和M条无向边。(N<=5000,M<=200000)
接下来M行每行包含三个整数Xi、Yi、Zi,表示有一条长度为Zi的无向边连接结点Xi、Yi
输出格式
输出包含一个数,即最小生成树的各边的长度之和;如果该图不连通则输出orz
输入输出样例
输入 #1
4 5
1 2 2
1 3 2
1 4 3
2 3 4
3 4 3
输出 #1
7
#include <bits/stdc++.h>
using namespace std;
const int maxn=200006;
int i,ans,n,m,cnt,ev,eu,fa[5005];
struct Edge{
int u,v,w;
}edge[maxn];
int find(int x)
{
while(x!=fa[x])
{
x=fa[x]=fa[fa[x]];//压缩路径
}
return x;
}
//并查集循环实现模板,及路径压缩,
bool cmp(Edge a,Edge b)
{
return a.w<b.w;
}
void kurska()
{
sort(edge,edge+m,cmp);
//将边的权值排序
for(i=0;i<m;i++)
{
eu=find(edge[i].u),ev=find(edge[i].v); //查看一条边的2点的连通状态
if(eu==ev) //如果2个点都已经连通了则跳过
continue;
ans+=edge[i].w;
fa[ev]=eu;
cnt++;
if(cnt==n-1) break;
}
}
int main()
{
cin>>n>>m;
for(i=0;i<=n;i++) //对父亲数组初始化
fa[i]=i;
for(i=0;i<m;i++)
{
cin>>edge[i].u>>edge[i].v>>edge[i].w;
}
kurska();
if(cnt==n-1) //如果满足边数==点数-1
cout<<ans;
else cout<<"orz"; //不连通
return 0;
}
prim算法:
1:先在图中任意选一个起点v1,放入集合VT
2:然后选取与VT中的点相连的未被选取过且使得边权最小的点,加入VT
3:然后重复步骤2直到所有点都被选取
再给个图吧
sum=5+2+1+4+10=22
再给一个参考别人的模板代码
#include<bits/stdc++.h>
using namespace std;
#define inf 123456789
#define maxn 5005
#define maxm 200005
struct edge
{
int v,w,next;
}e[maxm<<1];
//注意是无向图,开两倍数组
int head[maxn],dis[maxn],cnt,n,m,tot,now=1,ans;
//已经加入最小生成树的的点到没有加入的点的最短距离,比如说1和2号节点已经加入了最小生成树,那么dis[3]就等于min(1->3,2->3)
bool vis[maxn];
链式前向星加边
void add(int u,int v,int w)
{
e[++cnt].v=v;
e[cnt].w=w;
e[cnt].next=head[u];
head[u]=cnt;
//cout<<cnt<<endl;
}
int prim()
{
//先把dis数组附为极大值
for(int i=2;i<=n;++i)
{
dis[i]=inf;
}
//这里要注意重边,所以要用到min
for(int i=head[1];i;i=e[i].next)
{
dis[e[i].v]=min(dis[e[i].v],e[i].w);
}
while(++tot<n)//最小生成树边数等于点数-1
{
int minn=inf;//把minn置为极大值
vis[now]=1;//标记点已经走过
//枚举每一个没有使用的点
//找出最小值作为新边
//注意这里不是枚举now点的所有连边,而是1~n
for(int i=1;i<=n;++i)
{
if(!vis[i]&&minn>dis[i])
{
minn=dis[i];
now=i;
}
}
ans+=minn;
//枚举now的所有连边,更新dis数组
for(int i=head[now];i;i=e[i].next)
{
int v=e[i].v;
if(dis[v]>e[i].w&&!vis[v])
{
dis[v]=e[i].w;
}
}
}
return ans;
}
int main()
{
cin>>n>>m;
for(int i=1;i<=m;i++)
{
int u,v,w;
cin>>u>>v>>w;
add(u,v,w),add(v,u,w);
}
printf("%d",prim());
return 0;
}