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;
}