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