FZU 2087 统计树边(最小生成树)

Problem Description
在图论中,树:任意两个顶点间有且只有一条路径的图。

生成树:包含了图中所有顶点的一种树。

最小生成树:对于连通的带权图(连通网)G,其生成树也是带权的。生成树T各边的权值总和称为该树的权,权最小的生成树称为G的最小生成树(Minimum Spanning Tree)。最小生成树可简记为MST。

但是,对于一个图而言,最小生成树并不是唯一的。

现在,给你一个连通的有权无向图,图中不包含有自环和重边,你的任务就是寻找出有多少条边,它至少在一个最小生成树里。图保证连通。

Input
输入数据第一行包含一个整数T,表示测试数据的组数。对于每组测试数据:

第一行包含两个整数n,m(1

#include<cstdio>
#include<cstring>
#include<cmath>
#include<algorithm>
using namespace std;

int pre[1000005];
int fin(int x)
{
    if(pre[x]==x)
    {
        return x;
    }
    else
    {
        pre[x]=fin(pre[x]);//并查集的路径压缩
        return pre[x];
    }
}

void join(int x,int y)
{
    int t1=fin(x);
    int t2=fin(y);
    if(t1!=t2)
    pre[t1]=t2;
}

struct node
{
    int u;
    int v;
    int w;
}e[1100005];

int cmp(node x,node y)
{
    return x.w<y.w;
}

int main()
{
    int t,n,m;
    scanf("%d",&t);
    while(t--)
    {
        scanf("%d%d",&n,&m);
        memset(e,0,sizeof(e));
        for(int i=1; i<=n; i++)
        {
            pre[i]=i;
        }
        for(int i=1; i<=m; i++)
        {
            scanf("%d%d%d",&e[i].u,&e[i].v,&e[i].w);
        }
        sort(e+1,e+1+m,cmp);//对边按照权值进行排序
        int sum=0;
        for( int i=1,j; i<=m; i=j)
        {
            for(j=i; e[j].w==e[i].w; j++)
            {
                if(fin(e[j].u)!=fin(e[j].v))//如果不成环那么可以代替上条边
                {
                    sum++;
                }
            }
            for( j=i; e[j].w==e[i].w; j++)
            {
                if(fin(e[j].u)!=fin(e[j].v))//并查集的合并
                join(e[j].u,e[j].v);
            }
        }
        printf("%d\n",sum);
    }
    return 0;
}

你可能感兴趣的:(FZU)