题意
(T) 组数据,每组数据给定一个 (n) 个点,(m) 条边,可能含有重边自环的图,求出最小生成树的个数与边权和的乘积,对 (10^9+7) 取模。
( exttt{Data Range:}Tleq 100,2leq nleq 10^5,m=10^5)
题解
大家好,这题充分展现了我就是个 sb。
一见数据随机,立刻想到相同边权的边很少,立刻想到矩阵大小很小,立刻想到最小生成树计数,立刻想到 Matrix-Tree 定理。某些 Karry5307 的想像惟在这一层能够如此跃进。
直接进入正题,首先不能被题目中给出的最小生成树计数方法给带偏。
注意到边权在 (0sim 2^{64}-1) 范围内随机给定,所以我们有很大的把握认定最小生成树唯一,求出这个生成树的边权和即可。
代码
#include<bits/stdc++.h>
using namespace std;
typedef int ll;
typedef long long int li;
typedef unsigned long long int ull;
const ll MAXN=2e5+51,MOD=1e9+7;
struct EdgeForKruskal{
ll from,to;
ull dist;
inline bool operator <(const EdgeForKruskal &rhs)const
{
return this->dist<rhs.dist;
}
};
EdgeForKruskal ed[MAXN];
ll test,n,m,x,y;
ull z;
ll ffa[MAXN];
inline ll read()
{
register ll num=0,neg=1;
register char ch=getchar();
while(!isdigit(ch)&&ch!='-')
{
ch=getchar();
}
if(ch=='-')
{
neg=-1;
ch=getchar();
}
while(isdigit(ch))
{
num=(num<<3)+(num<<1)+(ch-'0');
ch=getchar();
}
return num*neg;
}
inline ll find(ll x)
{
return x==ffa[x]?x:ffa[x]=find(ffa[x]);
}
inline void merge(ll x,ll y)
{
ll fx=find(x),fy=find(y);
fx!=fy?ffa[fy]=fx:1;
}
inline ll Kruskal()
{
ll tott=0,res=0;
for(register int i=1;i<=m;i++)
{
if(find(ed[i].from)!=find(ed[i].to))
{
merge(ed[i].from,ed[i].to),res=(res+ed[i].dist%MOD)%MOD;
if(++tott==n-1)
{
break;
}
}
}
return tott==n-1?res:0;
}
namespace Maker{
ull k1,k2;
inline ull gen()
{
ull k3=k1,k4=k2;
k1=k4,k3^=k3<<23,k2=k3^k4^(k3>>17)^(k4>>26);
return k2+k4;
}
}
using namespace Maker;
inline void solve()
{
n=read(),m=read(),scanf("%llu%llu",&k1,&k2);
for(register int i=1;i<=n;i++)
{
ffa[i]=i;
}
for(register int i=1;i<=m;i++)
{
x=gen()%n+1,y=gen()%n+1,z=gen(),ed[i]=(EdgeForKruskal){x,y,z};
}
sort(ed+1,ed+m+1),printf("%d
",Kruskal());
}
int main()
{
test=read();
for(register int i=0;i<test;i++)
{
solve();
}
}