最小生成树模板(kruskal&&prim)

1.kruskal

  • 最小生成树能够保证首先是树(对于n个顶点的图只有条n - 1边),其次保证任意两个顶点之间都可达,再次保证这棵树的边权值之和为最小,但不能保证任意两点之间是最短路径;(最后求出来的路径长度是经过n个顶点的)
  • 最小生成树是到一群点(所有点)的路径代价和最小,是一个n - 1 条边的树,最短路是从一个点到另一个点的最短路径;

求最大生成树(所有点的路径代价之和最大)即把边从大到小排序

#include <bits/stdc++.h>
using namespace std;
const int N=1e3;
const int inf=0x3f3f3f3f;

struct node
{
    int u,v,w;
    bool operator <(const node &a)const
    {
        return w<a.w;
    }
}edge[N];

int father[N];
int n,m;

void init()
{
    for(int i=0;i<m;i++)
        father[i]=i;
}

int Find(int x)
{
    if(x==father[x])
        return x;
    return father[x]=Find(father[x]);
}

void Union(int x,int y)
{
    int tmpx=Find(x);
    int tmpy=Find(y);
    if(tmpx!=tmpy)
    {
        father[tmpx]=tmpy;
    }
}

int kruskal()
{
    sort(edge,edge+n);
    init();
    node now;
    int ans=0;
    for(int i=0;i<n;i++)
    {
        now=edge[i];
        if(Find(now.u)!=Find(now.v))
        {
            Union(now.u,now.v);
            ans+=now.w;
        }
    }
    return ans;
}

int main()
{
    while(scanf("%d%d",&n,&m)!=EOF&&n)
    {
        for(int i=0;i<n;++i)
        {
            scanf("%d%d%d",&edge[i].u,&edge[i].v,&edge[i].w);
        }
        int ans=kruskal();
        bool f=0;
        for(int i=2;i<=m;i++)
        {
            if(Find(1)!=Find(i))
            {
                f=1;
                break;
            }
        }
        if(f)
            cout<<"?"<<'\n';
        else
            cout<<ans<<'\n';
    }
    return 0;
}

2.prim

#include <bits/stdc++.h>
using namespace std;
const int N=1e3;
const int inf=0x3f3f3f3f;
int n,m;
int dis[N],vis[N],g[N][N];
int prim(int src)
{
    for(int i=1;i<=m;i++)
    {
        dis[i]=g[src][i];
    }
    int sum=0;
    vis[src]=1;
    dis[src]=0;
    for(int i=2;i<=m;i++)
    {
        int pos=-1;
        int minn=inf;
        for(int j=1;j<=m;j++)
        {
            if(dis[j]<minn&&!vis[j])
            {
                pos=j;
                minn=dis[j];
            }
        }
        if(pos==-1)
        {
            return -1;
        }
        sum+=minn;
        vis[pos]=1;
        for(int j=1;j<=m;j++)
        {
            if(!vis[j]&&dis[j]>g[pos][j])
            {
                dis[j]=g[pos][j];
            }
        }
    }
    return sum;
}
int main()
{
    int a,b,c;
    while(scanf("%d%d",&n,&m)!=EOF&&n)
    {
        memset(g,inf,sizeof(g));
        memset(dis,inf,sizeof(dis));
        memset(vis,0,sizeof(vis));
        for(int i=0;i<n;i++)
        {
            scanf("%d%d%d",&a,&b,&c);
            if(c<g[a][b])
                g[a][b]=g[b][a]=c;
        }
        int ans=prim(1);
        if(ans==-1)
            cout<<"?"<<'\n';
        else
            cout<<ans<<'\n';
    }
    return 0;
}

 

全部评论

相关推荐

就前几天旅游的时候,打开抖音就经常刷到这类视频:以前是高学历学生、老师、主持人,现在做着团播、擦边主播的工作,以及那些经过精心包装的“职业转型”故事——从铺天盖地的VLOG到所谓的“04年夜场工作日记”,这些内容在初中升学、高考放榜等关键时间节点持续发酵。可以说非常直接且精准地在潜移默化地影响着心智尚未成熟的青少年,使其对特殊行业逐渐脱敏。那我就想问了:某些传播公司、平台运营者甚至某些夜场的老板,你们究竟在传递怎样的价值观?点开那些视频,评论区里也是呈现明显的两极分化:一种是​​经济下行论​​:“现在就业市场已经艰难到这种程度了吗?”​​一种是事实反驳派​​:这些创作者往往拥有名校背景,从事着...
牛客刘北:被环境教育的,为了能拿到足够的钱养活自己,不甘心也得甘心,现在的短视频传播的思想的确很扭曲,但是很明显,互联网玩上一年你就能全款提A6,但你全心全意不吃不喝工作一年未必能提A6,但是在高考中考出现这个的确很扭曲,在向大家传播“不上学,玩互联网也可以轻松年入百万”,不是人变了,是社会在变
预测一下26届秋招形势
点赞 评论 收藏
分享
每晚夜里独自颤抖:这个在牛客不是老熟人了吗
点赞 评论 收藏
分享
点赞 评论 收藏
分享
评论
点赞
收藏
分享

创作者周榜

更多
牛客网
牛客网在线编程
牛客网题解
牛客企业服务