BZOJ 1016 [JSOI2008]最小生成树计数 ——Matrix-Tree定理
考虑从小往大加边,然后把所有联通块的生成树个数计算出来。
然后把他们缩成一个点,继续添加下一组。
最后乘法原理即可。
写起来很恶心
#include <queue>
#include <cmath>
#include <vector>
#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>
using namespace std; #define F(i,j,k) for (int i=j;i<=k;++i)
#define D(i,j,k) for (int i=j;i>=k;--i)
#define maxn 1005
#define eps 1e-6
const int md=31011;
vector <int> v,to[maxn];
queue <int> q;
struct Edge{int u,v,w;}a[maxn];
int n,m,fa[maxn];
int b[maxn][maxn],inv[maxn];
int vcnt,du[maxn],list[maxn],vis[maxn];
bool cmp(Edge x,Edge y)
{return x.w<y.w;}
int gf(int k)
{if (fa[k]==k) return k; else return fa[k]=gf(fa[k]);}
int gauss(int n)
{
F(i,1,n) F(j,1,n) b[i][j]%=md;
int ret=1;
for (int i=1;i<n;++i)
{
for (int j=i+1;j<n;++j)
while (b[j][i])
{
int t=b[i][i]/b[j][i];
for (int k=i;k<n;++k)
b[i][k]=(b[i][k]-b[j][k]*t+md)%md;
for (int k=i;k<n;++k)
swap(b[i][k],b[j][k]);
ret=-ret;
}
if (b[i][i]==0) return 0;
ret=ret*b[i][i]%md;
}
return abs((ret+md)%md);
}
int main()
{
scanf("%d%d",&n,&m);
F(i,1,n) fa[i]=i;
F(i,1,m){scanf("%d%d%d",&a[i].u,&a[i].v,&a[i].w);}
sort(a+1,a+m+1,cmp);
int now=1,ans=1;
while (now<=m)
{
int l=now,r=now;
vcnt=0;
memset(du,0,sizeof du);
F(i,1,n) to[i].clear();
while (a[r+1].w==a[r].w) r++;
now=r+1;
F(i,l,r)
{
int fl=gf(a[i].u),fr=gf(a[i].v);
to[fl].push_back(fr);
to[fr].push_back(fl);
if (fl!=fr) du[fl]++,du[fr]++;
}
memset(vis,0,sizeof vis);
F(i,1,n)
if (du[i]&&!vis[i])
{
v.clear();
memset(b,0,sizeof b);
memset(inv,0,sizeof inv);
q.push(i);inv[i]=1;vis[i]=1;
while (!q.empty())
{
int x=q.front();v.push_back(x);q.pop();
for (int j=0;j<to[x].size();++j)
if (!vis[to[x][j]])
q.push(to[x][j]),inv[to[x][j]]=1,vis[to[x][j]]=1;
}
for (int j=0;j<v.size();++j) list[v[j]]=j+1;
for (int j=0;j<v.size();++j)
for (int k=0;k<to[v[j]].size();++k)
if (inv[to[v[j]][k]])
{
b[list[v[j]]][list[v[j]]]++,b[list[v[j]]][list[to[v[j]][k]]]--;
}
ans*=gauss(v.size());
ans%=md;
}
F(i,l,r)
{
int fl=gf(a[i].u),fr=gf(a[i].v);
if (fl!=fr){fa[fl]=fr;}
}
}
int cnt=0;
F(i,1,n) if (fa[i]==i)
{
cnt++;
if (cnt==2) {printf("0\n"); return 0;}
}
printf("%d\n",ans);
}
BZOJ 1016 [JSOI2008]最小生成树计数 ——Matrix-Tree定理的更多相关文章
- BZOJ.1016.[JSOI2008]最小生成树计数(Matrix Tree定理 Kruskal)
题目链接 最小生成树有两个性质: 1.在不同的MST中某种权值的边出现的次数是一定的. 2.在不同的MST中,连接完某种权值的边后,形成的连通块的状态是一样的. \(Solution1\) 由这两个性 ...
- bzoj 1016 [JSOI2008]最小生成树计数——matrix tree(相同权值的边为阶段缩点)(码力)
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=1016 就是缩点,每次相同权值的边构成的联通块求一下matrix tree.注意gauss里的 ...
- BZOJ 1016: [JSOI2008]最小生成树计数( kruskal + dfs )
不同最小生成树中权值相同的边数量是一定的, 而且他们对连通性的贡献是一样的.对权值相同的边放在一起(至多10), 暴搜他们有多少种方案, 然后乘法原理. ----------------------- ...
- [BZOJ 1016] [JSOI2008] 最小生成树计数 【DFS】
题目链接:BZOJ - 1016 题目分析 最小生成树的两个性质: 同一个图的最小生成树,满足: 1)同一种权值的边的个数相等 2)用Kruscal按照从小到大,处理完某一种权值的所有边后,图的连通性 ...
- [BZOJ]1016 JSOI2008 最小生成树计数
最小生成树计数 题目描述 现在给出了一个简单无向加权图.你不满足于求出这个图的最小生成树,而希望知道这个图中有多少个不同的最小生成树.(如果两颗最小生成树中至少有一条边不同,则这两个最小生成树就是不同 ...
- bzoj 1016: [JSOI2008]最小生成树计数【dfs+克鲁斯卡尔】
有一个性质就是组成最小生成树总边权值的若干边权总是相等的 这意味着按边权排序后在权值相同的一段区间内的边能被选入最小生成树的条数是固定的 所以先随便求一个最小生成树,把每段的入选边数记录下来 然后对于 ...
- 【BZOJ 1016】 1016: [JSOI2008]最小生成树计数 (DFS|矩阵树定理)
1016: [JSOI2008]最小生成树计数 Description 现在给出了一个简单无向加权图.你不满足于求出这个图的最小生成树,而希望知道这个图中有多少个不同的最小生成树.(如果两颗最小生成树 ...
- 1016: [JSOI2008]最小生成树计数
1016: [JSOI2008]最小生成树计数 Time Limit: 1 Sec Memory Limit: 162 MBSubmit: 6200 Solved: 2518[Submit][St ...
- @总结 - 7@ 生成树计数 —— matrix - tree 定理(矩阵树定理)与 prüfer 序列
目录 @0 - 参考资料@ @0.5 - 你所需要了解的线性代数知识@ @1 - 矩阵树定理主体@ @证明 part - 1@ @证明 part - 2@ @证明 part - 3@ @证明 part ...
随机推荐
- NSString+TimeCategory
NSString+TimeCategory.h //------------------------------------------------ #import <foundation fo ...
- C#入门(3)
C#入门(3) Delegates, Events, Lambda Expressions 最早的windows是使用c风格的函数指针来进行callback的,但是这样仅仅传递了一个内存中的地址,无法 ...
- UVA1663 Purifying Machine (匈牙利算法,二分图最大匹配)
模版集合个数减少是因为匹配串集合中没被匹配过的一对串匹配了.所以就是找一个二分图最大匹配. 因为集合X和Y是不好分开的,但是可以直接跑,两个集合都会跑一遍,所以一个匹配会被算两次,返回的时候除以2就行 ...
- kubernetes添加不了google apt-key
转自icepoint的博客 key来源 我的百度云盘 密码:v3wo 下载kube_apt_key.gpg到本地,上传到服务器后执行下面的命令 apt-get update && ap ...
- hihoCode-1043-完全背包
我们定义:best(i,x)代表i件以前的物品已经决定好选择多少件,并且在剩余奖券x的情况下的最优解. 我们可以考虑最后一步,是否再次选择i物品,在不超过持有奖券总额的情况下.上面的第二个式子的k是大 ...
- HDU-1312-Black and Red
这题其实和POJ的1979是同一道题,当时POJ使用cin写的,所以读入的时候,就很正确. 这次用scanf读入的时候,就出现了问题,我们在读完宽高之后,要用getchar吸收掉回车,然后每行末尾的回 ...
- (32)zabbix分布式监控proxy vs nodes
概述 zabbix为IT基础设施提供有效和可用的分布式监控,zabbix提供了两种解决方案,分别为:proxy和nodes.proxy代替zabbix server在本地检索数据,然后提交给zabbi ...
- Python数据分析库之pandas,你该这么学!No.1
写这个系列背后的故事 咦,面试系列的把基础部分都写完啦,哈哈答,接下来要弄啥嘞~ pandas吧 外国人开发的 翻译成汉语叫 熊猫 厉害厉害,很接地气 一个基于numpy的库 干啥的? 做数据分析用的 ...
- 爬虫练习四:爬取b站番剧字幕
由于个人经常在空闲时间在b站看些小视频欢乐一下,这次就想到了爬取b站视频的弹幕. 这里就以番剧<我的妹妹不可能那么可爱>第一季为例,抓取这一番剧每一话对应的弹幕. 1. 分析页面 这部番剧 ...
- int main(int argc,char *argv[])的具体含义
int main(int argc,char * argv[]) argv为指针的指针 argc为整数 char **argv or: char *argv[] or: char argv[][] m ...