考虑从小往大加边,然后把所有联通块的生成树个数计算出来。

然后把他们缩成一个点,继续添加下一组。

最后乘法原理即可。

写起来很恶心

#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定理的更多相关文章

  1. BZOJ.1016.[JSOI2008]最小生成树计数(Matrix Tree定理 Kruskal)

    题目链接 最小生成树有两个性质: 1.在不同的MST中某种权值的边出现的次数是一定的. 2.在不同的MST中,连接完某种权值的边后,形成的连通块的状态是一样的. \(Solution1\) 由这两个性 ...

  2. bzoj 1016 [JSOI2008]最小生成树计数——matrix tree(相同权值的边为阶段缩点)(码力)

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=1016 就是缩点,每次相同权值的边构成的联通块求一下matrix tree.注意gauss里的 ...

  3. BZOJ 1016: [JSOI2008]最小生成树计数( kruskal + dfs )

    不同最小生成树中权值相同的边数量是一定的, 而且他们对连通性的贡献是一样的.对权值相同的边放在一起(至多10), 暴搜他们有多少种方案, 然后乘法原理. ----------------------- ...

  4. [BZOJ 1016] [JSOI2008] 最小生成树计数 【DFS】

    题目链接:BZOJ - 1016 题目分析 最小生成树的两个性质: 同一个图的最小生成树,满足: 1)同一种权值的边的个数相等 2)用Kruscal按照从小到大,处理完某一种权值的所有边后,图的连通性 ...

  5. [BZOJ]1016 JSOI2008 最小生成树计数

    最小生成树计数 题目描述 现在给出了一个简单无向加权图.你不满足于求出这个图的最小生成树,而希望知道这个图中有多少个不同的最小生成树.(如果两颗最小生成树中至少有一条边不同,则这两个最小生成树就是不同 ...

  6. bzoj 1016: [JSOI2008]最小生成树计数【dfs+克鲁斯卡尔】

    有一个性质就是组成最小生成树总边权值的若干边权总是相等的 这意味着按边权排序后在权值相同的一段区间内的边能被选入最小生成树的条数是固定的 所以先随便求一个最小生成树,把每段的入选边数记录下来 然后对于 ...

  7. 【BZOJ 1016】 1016: [JSOI2008]最小生成树计数 (DFS|矩阵树定理)

    1016: [JSOI2008]最小生成树计数 Description 现在给出了一个简单无向加权图.你不满足于求出这个图的最小生成树,而希望知道这个图中有多少个不同的最小生成树.(如果两颗最小生成树 ...

  8. 1016: [JSOI2008]最小生成树计数

    1016: [JSOI2008]最小生成树计数 Time Limit: 1 Sec  Memory Limit: 162 MBSubmit: 6200  Solved: 2518[Submit][St ...

  9. @总结 - 7@ 生成树计数 —— matrix - tree 定理(矩阵树定理)与 prüfer 序列

    目录 @0 - 参考资料@ @0.5 - 你所需要了解的线性代数知识@ @1 - 矩阵树定理主体@ @证明 part - 1@ @证明 part - 2@ @证明 part - 3@ @证明 part ...

随机推荐

  1. input禁止显示用户输入历史记录

    input标签中加上属性autocomplete="off"

  2. 这么大一座Azure“图书馆”,你竟没有发现…

    为避免被叫做「伸手党」,很多技术人员早已养成遇到问题上网搜的好习惯. 然而…… 同一个概念,搜到两个相互矛盾的解释,以谁的为准? 想查找某个 API 的用法,搜索结果数十万条,怎样筛选出最有价值的? ...

  3. 一次性删除数据库所有表和所有存储过程 SQL语句

    一次性删除数据库所有表和所有存储过程 SQL语句 今天转移数据库数据,需要把数据库原来的表和存储过程清空.删除所有的表:如果由于外键约束删除table失败,则先删除所有约束: --/第1步****** ...

  4. COGS 696. [IOI1996][USACO 2.3] 最长前缀

    ★   输入文件:prefix.in   输出文件:prefix.out   简单对比时间限制:1 s   内存限制:128 MB 描述 USACO 2.3.1 IOI96 在生物学中,一些生物的结构 ...

  5. 覆盖alert对话框-自制Jquery.alert插件

    Javascript 代码: (function ($) { 'use strict'; window.alert = $.alert = function (msg) { var defaultOp ...

  6. 11gR2新特性---Gpnp守护进程

    在这篇文章中,我们会对11gR2 新的守护进程(资源名称ora.gpnpd)进行介绍,其中包含的gpnp的功能,启动顺序和基本的诊断方法. gpnp全称为grid plug and play,该组件的 ...

  7. elastic-job lite 编程实战经验

    (继续贴一篇之前写的经验案例) elastic-job lite 编程实战经验 其实这是一次失败的项目,虽然最后还是做出来了,但是付出了很大代价.并且需要较深入的踩坑改造elastic-job,导致代 ...

  8. python基础面试题整理---从零开始 每天十题(04)

    一.Q:如何用Python来进行查询和替换一个文本字符串? A:可以使用sub()方法来进行查询和替换,sub方法的格式为:sub(replacement, string[, count=0]) re ...

  9. ext笔记

    命名   The top-level namespaces and the actual class names should be CamelCased. Everything else shoul ...

  10. 如何正确入门Windows系统下驱动开发领域?

    [作者]猪头三个人网站 :http://www.x86asm.com/ [序言]很多人都对驱动开发有兴趣,但往往找不到正确的学习方式.当然这跟驱动开发的本土化资料少有关系.大多学的驱动开发资料都以英文 ...