bzoj1016:[JSOI2008]最小生成树计数

思路:模拟kruskal的过程,可以发现对于所有权值相同的边,有很多种选择的方案,而且权值不同的边并不会相互影响,因为先考虑权值较小的边,权值比当前权值大的边显然不在考虑范围之内,而权值比当前权值小的边所组成的连通块已经经过缩点变成一个点了,因此处理权值相同的所有边可以看成是一个阶段,最后的答案也就是所有阶段的答案的乘积(乘法原理)。

那么如何来处理权值相同的方案数呢,同样还是考虑kruskal的过程,因为权值相同的边可能会组成很多个连通块,且连通块之间互不影响,因此只考虑单个连通块即可(还是乘法原理),如果一条边所连接的两个点不在一个连通块内,那么就把这条边算进答案,那么对于一个连通块kruskal的过程显然要让所有点连通,且所选的边恰好构成了一棵树,那么这样问题就转化成了如何求生成树的数量,利用matrix-tree定理,用高斯消元求解kirchhoff矩阵即可。

还有最后不要忘了判图中没有最小生成树的情况(图不连通)

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<vector>
using namespace std;
#define maxm 1005
#define maxn 105
#define mod 31011

int n,m,cnt,ans=1;
int fa[maxn],pos[maxn];
int K[maxn][maxn];
bool vis[maxn];

vector<int> v[maxn];

struct edge{
	int from,to,val;
	bool operator <(const edge &a)const{return val<a.val;}
}e[maxm];

inline int read(){
	int x=0;char ch=getchar();
	for (;ch<‘0‘||ch>‘9‘;ch=getchar());
	for (;ch>=‘0‘&&ch<=‘9‘;ch=getchar()) x=x*10+ch-‘0‘;
	return x;
}

int find(int x){return fa[x]==x?x:fa[x]=find(fa[x]);}

int gauss(){
	int t,n=cnt-1,ans=1,f=1;
	for (int i=1;i<=n;i++)
		for (int j=1;j<=n;j++)
			K[i][j]%=mod;
	for (int i=1;i<n;i++){
		for (t=i;t<=n;t++) if (K[t][i]) break;if (t>n) return 0;
		if (t!=i){for (int j=1;j<=n;j++) swap(K[i][j],K[t][j]);f=-f;}
		for (int j=i+1;j<=n;j++)
			for (;K[j][i];){
				int t=K[i][i]/K[j][i];
				for (int k=i;k<=n;k++) K[i][k]=(K[i][k]-t*K[j][k])%mod;
				for (int k=i;k<=n;k++) swap(K[i][k],K[j][k]);
				f=-f;
			}
	}
	for (int i=1;i<=n;i++) ans=1ll*ans*K[i][i]%mod;
	return ans*f;
}

void dfs(int x,int num){
	K[num][num]=v[x].size(),pos[x]=num;
	for (unsigned int i=0;i<v[x].size();i++){
		vis[v[x][i]]=0;
		if (!pos[v[x][i]]) pos[v[x][i]]=++cnt,K[num][pos[v[x][i]]]--,dfs(v[x][i],cnt);
		else K[num][pos[v[x][i]]]--;
	}
}

int main(){
	n=read(),m=read();
	for (int i=1;i<=m;i++) e[i].from=read(),e[i].to=read(),e[i].val=read();
	for (int i=1;i<=n;i++) fa[i]=i;
	sort(e+1,e+m+1);
	for (int i=1;i<=m+1;i++){
		int ck=find(1);
		for (int j=2;j<=n;j++) if (find(j)!=ck){ck=0;break;}
		if (ck) break;
		int x=find(e[i].from),y=find(e[i].to);
		if (x!=y) vis[x]=1,vis[y]=1,v[x].push_back(y),v[y].push_back(x);
		if (e[i].val!=e[i+1].val){
			for (int j=1;j<=n;j++)
				if (vis[j]){
					for (int a=1;a<=cnt;a++)
						for (int b=1;b<=cnt;b++)
							K[a][b]=0;
					memset(pos,0,sizeof(pos));
					vis[j]=0,cnt=1,dfs(j,cnt);
					ans=1ll*ans*gauss()%mod;
				}
			for (int j=1;j<=n;j++){
				for (unsigned int k=0;k<v[j].size();k++){
					int x=find(j),y=find(v[j][k]);
					if (x!=y) fa[x]=y;
				}
				v[j].clear();
			}
		}
	}
	int check=find(1);
	for (int i=2;i<=n;i++) if (find(i)!=check){check=0;break;}
	printf("%d\n",check?ans:0);
	return 0;
}
时间: 2024-10-16 04:07:59

bzoj1016:[JSOI2008]最小生成树计数的相关文章

bzoj1016: [JSOI2008]最小生成树计数(kruskal+dfs)

1016: [JSOI2008]最小生成树计数 题目:传送门 题解: 神题神题%%% 据说最小生成树有两个神奇的定理: 1.权值相等的边在不同方案数中边数相等  就是说如果一种方案中权值为1的边有n条    那么在另一种方案中权值为1的边也一定有n条 2.如果边权为1的边连接的点是x1,x2,x3   那么另一种方案中边权为1的边连接的也一定是x1,x2,x3  如果知道了这两条定理那就很好做了啊: 因为等权边的条数一定,那么我们就可以预处理求出不同边权的边的条数 题目很人道的保证了边权相同的边

[BZOJ1016][JSOI2008]最小生成树计数(结论题)

题目:http://www.lydsy.com:808/JudgeOnline/problem.php?id=1016 分析: 首先有个性质:如果边集E.E'都可以表示一个图G的最小生成树(当然E和E’的元素个数肯定一样),那么某确定权值的边在E中出现的次数==在E‘中出现的次数 简单证明一下: 按照Kruskal算法的流程来想,首先我们知道Kruskal求一个最小生成树是正确的,那么不同的最小生成树会怎么产生呢?当然是Kruskal选择权值相同的边的顺序,很有可能选择权值相同边的顺序不同导致后

bzoj1016 [JSOI2008]最小生成树计数

Description 现在给出了一个简单无向加权图.你不满足于求出这个图的最小生成树,而希望知道这个图中有多少个不同的最小生成树.(如果两颗最小生成树中至少有一条边不同,则这两个最小生成树就是不同的).由于不同的最小生成树可能很多,所以你只需要输出方案数对31011的模就可以了. Input 第一行包含两个数,n和m,其中1<=n<=100; 1<=m<=1000; 表示该无向图的节点数和边数.每个节点用1~n的整数编号.接下来的m行,每行包含两个整数:a, b, c,表示节点a

BZOJ1016 JSOI2008 最小生成树计数 生成树+DFS

题意:求最小生成树的方案数,保证每个边权出现的次数小于十次. 题解:首先我们需要知道:一张图对于一个确定的边权,在任意最小生成树中出现的次数是相同的(请不要问我为什么QAQ).所以我们先求出每一种边权在MST中出现的次数,然后枚举每一个边权,暴力看取哪些边可以组出一颗MST,复杂度O(M*2^10*M/10) #include <cstdio> #include <cstring> #include <cstdlib> #include <iostream>

[BZOJ1016] [JSOI2008] 最小生成树计数 (Kruskal)

Description 现在给出了一个简单无向加权图.你不满足于求出这个图的最小生成树,而希望知道这个图中有多少个不同的最小生成树.(如果两颗最小生成树中至少有一条边不同,则这两个最小生成树就是不同的).由于不同的最小生成树可能很多,所以你只需要输出方案数对31011的模就可以了. Input 第一行包含两个数,n和m,其中1<=n<=100; 1<=m<=1000; 表示该无向图的节点数和边数.每个节点用1~n的整数编号.接下来的m行,每行包含两个整数:a, b, c,表示节点a

【最小生成树】BZOJ1016: [JSOI2008]最小生成树计数

Description 现在给出了一个简单无向加权图.你不满足于求出这个图的最小生成树,而希望知道这个图中有多少个不同的最小生成树.(如果两颗最小生成树中至少有一条边不同,则这两个最小生成树就是不同的).由于不同的最小生成树可能很多,所以你只需要输出方案数对31011的模就可以了. Solution 把所有边权相同的视为边组,每一组边组在最小生成树的条数是固定的,对连通性的贡献也是固定的.(证明可以看http://www.cnblogs.com/Fatedayt/archive/2012/05/

【kruscal】【最小生成树】【搜索】bzoj1016 [JSOI2008]最小生成树计数

不用Matrix-tree定理什么的,一边kruscal一边 对权值相同的边 暴搜即可.将所有方案乘起来. 1 #include<cstdio> 2 #include<algorithm> 3 using namespace std; 4 int n,m; 5 struct Disjoint_Set 6 { 7 int fa[101],rank[101]; 8 void init(){for(int i=1;i<=n;i++) fa[i]=i;} 9 int findroot

【JSOI2008】【BZOJ1016】最小生成树计数

我就爱写矩阵树定理!!! 就不写暴力!!! 1016: [JSOI2008]最小生成树计数 Time Limit: 1 Sec Memory Limit: 162 MB Submit: 3584 Solved: 1429 [Submit][Status][Discuss] Description 现在给出了一个简单无向加权图.你不满足于求出这个图的最小生成树,而希望知道这个图中有多少个不同的最小生成树.(如果两颗最小生成树中至少有一条边不同,则这两个最小生成树就是不同的).由于不同的最小生成树可

JSOI2008 最小生成树计数

题解: 最小生成树的两个性质: 1.边权相等的边的个数一定. 2.做完边权为w的所有边时,图的连通性相同. 证明: 1.边权相等的边的个数不一样的话就不会都同时是最小生成树了. 2.假设每种方法的做完边权为w的连通性不同,那么假设i边和j边没有同时被选,那么我们完全可以在一种方案中加入i边(或j边),使得连通性增强,而后面费用更大的边用的更少,这样与这是最小生成树矛盾.于是,命题得证. 代码:不知为何,下面程序有bug,什么时候再回来A掉…… 1 type node1=record 2 x,y,

BZOJ 题目1016: [JSOI2008]最小生成树计数(Kruskal+Matrix_Tree)

1016: [JSOI2008]最小生成树计数 Time Limit: 1 Sec  Memory Limit: 162 MB Submit: 3569  Solved: 1425 [Submit][Status][Discuss] Description 现在给出了一个简单无向加权图.你不满足于求出这个图的最小生成树,而希望知道这个图中有多少个不同的最小生成树.(如果两颗最小生成树中至少有一条边不同,则这两个最小生成树就是不同的).由于不同的最小生成树可能很多,所以你只需要输出方案数对3101