●BZOJ 4596 [Shoi2016]黑暗前的幻想乡

题链:

http://www.lydsy.com/JudgeOnline/problem.php?id=4596

题解:

容斥,矩阵树定理,矩阵行列式

先说说容斥:(一共有 N-1个公司)
令 f[i] 表示选出 (N-1)-i 个公司来修路(即有i个公司一定不修),且不管每个公司只能修一条路这一限制的方案数。
那么 答案 ANS=0个公司不修的方案数 - 1个公司不修的方案数 +2个公司不修的翻案数 ...
即 ANS= f[0] - f[1] +f[2] - ... + (-1)i*f[i]
f[i]的求法呢,就是先O(2N)枚举公司集合情况,
然后用矩阵树定理 O(N3) 求出当前情况下的生成树方案数。
另外再本题中,在构造上三角矩阵用以求行列式时,
既要避免小数,还要不影响原矩阵的行列式的值,
所以采用辗转相除的高斯消元法去构造上三角矩阵。复杂度多一个(logN)
由矩阵行列式的性质可知,这样辗转相除的高斯消元法不会影响行列式的绝对值,
只会影响符号位的正负,所以统计一下正负号的变化就好了。
(还不会矩阵树定理,看看这里,入门一波。)(还没写哈…)
所以总的时间复杂度为 O(2N*N3*logN)。
(都是这个复杂度,不晓得为什么我的代码跑到这么慢,都垫底了......)

代码:

#include<cstdio>
#include<cstring>
#include<iostream>
#define add(x,y) (((1ll*(x)+(y))%mod+mod)%mod)
#define mul(x,y) (((1ll*(x)*(y))%mod+mod)%mod)
#define filein(x) freopen(#x".in","r",stdin)
#define fileout(x) freopen(#x".out","w",stdout)
using namespace std;
const int mod=1000000007;
struct Matrix{
	int Val[20][20],*X[20],R,C;
	void Init(int r,int c){//r==c
		R=r; C=c;
		memset(Val,0,sizeof(Val));
		for(int i=1;i<=R;i++) X[i]=Val[i];
	}
	void Modify(int r,int c,int v){
		X[r][c]=add(X[r][c],v);
	}
	void operator =(const Matrix &rtm){
		Init(rtm.R,rtm.C);
		for(int i=1;i<=R;i++)
			for(int j=1;j<=C;j++)
				Val[i][j]=rtm.X[i][j];
	}
	Matrix operator -(const Matrix & rtm) const{
		Matrix now; now=*this;
		for(int i=1;i<=R;i++)
			for(int j=1;j<=C;j++)
				now.X[i][j]=add(now.X[i][j],-rtm.X[i][j]);
		return now;
	}
	void Gauss_Euclidean(int p,int &ti){//形成上三角矩阵
		if(p==R-1) return;
		if(!X[p][p])
			for(int i=p+1;i<R;i++) if(X[i][p]){
				swap(X[i],X[p]); ti++; break;
			}
		if(!X[p][p]) return;
		for(int i=p+1;i<R;i++){
			while(X[i][p]){
				int t=X[p][p]/X[i][p];
				for(int j=p;j<R;j++)
					X[p][j]=add(X[p][j],-mul(X[i][j],t));
				swap(X[p],X[i]); ti++;
			}
		}
		Gauss_Euclidean(p+1,ti);
	}
	int Determinant(){
		int ti=0,ans=1;
		Gauss_Euclidean(1,ti);
		for(int i=1;i<R;i++) ans=mul(ans,X[i][i]);
		if(ti&1) ans=mul(ans,-1);
		return ans;
	}
	void print(){
		for(int i=1;(i!=1?printf("\n"):0),i<=R;i++)
			for(int j=1;j<=R;j++)
				printf("%d ",X[i][j]);
	}
}A,B,K;
int cpy[20][500];
int ANS,N,tmp;
void dfs(int p,int num){
	if(p>=N) return;
	//选
	for(int i=1,a,b;i<=2*cpy[p][0];i+=2){
		a=cpy[p][i]; b=cpy[p][i+1];
		A.Modify(a,a,1); A.Modify(b,b,1);
		B.Modify(a,b,1); B.Modify(b,a,1);
	}
	K=A-B; tmp=K.Determinant();
	if(((N-1)-(num+1))&1) tmp=mul(tmp,-1);
	ANS=add(ANS,tmp);
	dfs(p+1,num+1);
	//不选
	for(int i=1,a,b;i<=2*cpy[p][0];i+=2){
		a=cpy[p][i]; b=cpy[p][i+1];
		A.Modify(a,a,-1); A.Modify(b,b,-1);
		B.Modify(a,b,-1); B.Modify(b,a,-1);
	}
	dfs(p+1,num);
}
int main()
{
	scanf("%d",&N);
	A.Init(N,N); B.Init(N,N);
	for(int i=1;i<=N-1;i++){
		scanf("%d",&cpy[i][0]);
		for(int j=1;j<=2*cpy[i][0];j++)
			scanf("%d",&cpy[i][j]);
	}
	dfs(1,0);
	printf("%d",ANS);
	return 0;
}
时间: 2024-08-20 05:28:40

●BZOJ 4596 [Shoi2016]黑暗前的幻想乡的相关文章

BZOJ 4596: [Shoi2016]黑暗前的幻想乡

4596: [Shoi2016]黑暗前的幻想乡 Time Limit: 20 Sec  Memory Limit: 256 MBSubmit: 408  Solved: 232[Submit][Status][Discuss] Description 四年一度的幻想乡大选开始了,最近幻想乡最大的问题是很多来历不明的妖 怪涌入了幻想乡,扰乱了幻想乡昔日的秩序.但是幻想乡的建制派妖怪(人类) 博丽灵梦和八云紫等人整日高谈所有妖怪平等,幻想乡多元化等等,对于幻想乡 目前面临的种种大问题却给不出合适的解

bzoj4596[Shoi2016]黑暗前的幻想乡 Matrix定理+容斥原理

4596: [Shoi2016]黑暗前的幻想乡 Time Limit: 20 Sec  Memory Limit: 256 MBSubmit: 464  Solved: 264[Submit][Status][Discuss] Description 四年一度的幻想乡大选开始了,最近幻想乡最大的问题是很多来历不明的妖 怪涌入了幻想乡,扰乱了幻想乡昔日的秩序.但是幻想乡的建制派妖怪(人类) 博丽灵梦和八云紫等人整日高谈所有妖怪平等,幻想乡多元化等等,对于幻想乡 目前面临的种种大问题却给不出合适的解

bzoj4596/luoguP4336 [SHOI2016]黑暗前的幻想乡(矩阵树定理,容斥)

bzoj4596/luoguP4336 [SHOI2016]黑暗前的幻想乡(矩阵树定理,容斥) bzoj Luogu 题解时间 看一看数据范围,求生成树个数毫无疑问直接上矩阵树定理. 但是要求每条边都属于不同公司就很难直接实现. 按套路上容斥: 如果直接将几个公司的修路列表加进矩阵里的话,求出来的是"只使用"这些边的生成树个数. 很明显上容斥之后就会直接变成"只使用"且"每个都被使用"的个数. 正好符合题目要求的生成树的n-1条边分属于n-1个公

[ZJOI2016]小星星&amp;[SHOI2016]黑暗前的幻想乡(容斥)

这两道题思路比较像,所以把他们放到一块. [ZJOI2016]小星星 题目描述 小Y是一个心灵手巧的女孩子,她喜欢手工制作一些小饰品.她有n颗小星星,用m条彩色的细线串了起来,每条细线连着两颗小星星. 有一天她发现,她的饰品被破坏了,很多细线都被拆掉了.这个饰品只剩下了n-1条细线,但通过这些细线,这颗小星星还是被串在一起,也就是这些小星星通过这些细线形成了树.小Y找到了这个饰品的设计图纸,她想知道现在饰品中的小星星对应着原来图纸上的哪些小星星.如果现在饰品中两颗小星星有细线相连,那么要求对应的

P4336 [SHOI2016]黑暗前的幻想乡

题目 P4336 [SHOI2016]黑暗前的幻想乡 做法 每种颜色恰好一条边,有点难处理啊 根据套路,数据范围这么小,容斥一下所有的情况就可以了 对每种颜色进行状压,表这次只能选这些颜色,做\(n-1\)的时候会加上多余的东西,就减去\(n-2......\) 剩下的交给矩阵树,这题有模数,高斯消元的时候逆元 My complete code #include<cstring> #include<string> #include<iostream> #include&

BZOJ4596: [Shoi2016]黑暗前的幻想乡

Description 四年一度的幻想乡大选开始了,最近幻想乡最大的问题是很多来历不明的妖 怪涌入了幻想乡,扰乱了幻想乡昔日的秩序.但是幻想乡的建制派妖怪(人类) 博丽灵梦和八云紫等人整日高谈所有妖怪平等,幻想乡多元化等等,对于幻想乡 目前面临的种种大问题却给不出合适的解决方案. 风间幽香是幻想乡里少有的意识到了问题的严重性的大妖怪.她这次勇敢的 站了出来参加幻想乡大选.提出包括在幻想乡边境建墙(并让人类出钱),大力 开展基础设施建设挽回失业率等一系列方案,成为了大选年出人意料的黑马并顺 利的当

[SHOI2016]黑暗前的幻想乡

Description 四年一度的幻想乡大选开始了,最近幻想乡最大的问题是很多来历不明的妖 怪涌入了幻想乡,扰乱了幻想乡昔日的秩序.但是幻想乡的建制派妖怪(人类) 博丽灵梦和八云紫等人整日高谈所有妖怪平等,幻想乡多元化等等,对于幻想乡 目前面临的种种大问题却给不出合适的解决方案. 风间幽香是幻想乡里少有的意识到了问题的严重性的大妖怪.她这次勇敢的 站了出来参加幻想乡大选.提出包括在幻想乡边境建墙(并让人类出钱),大力 开展基础设施建设挽回失业率等一系列方案,成为了大选年出人意料的黑马并顺 利的当

【bzoj4596】[Shoi2016]黑暗前的幻想乡

Description 四年一度的幻想乡大选开始了,最近幻想乡最大的问题是很多来历不明的妖 怪涌入了幻想乡,扰乱了幻想乡昔日的秩序.但是幻想乡的建制派妖怪(人类) 博丽灵梦和八云紫等人整日高谈所有妖怪平等,幻想乡多元化等等,对于幻想乡 目前面临的种种大问题却给不出合适的解决方案. 风间幽香是幻想乡里少有的意识到了问题的严重性的大妖怪.她这次勇敢的 站了出来参加幻想乡大选.提出包括在幻想乡边境建墙(并让人类出钱),大力 开展基础设施建设挽回失业率等一系列方案,成为了大选年出人意料的黑马并顺 利的当

「SHOI2016」黑暗前的幻想乡

题目链接 戳我 \(Describe\) \(n?1\)个公司,每个公司能修一些边,求每条边都让不同的公司来修的生成树的方案数 \(Solution\) 这道题很明显容斥.答案就是:所有都选的生成树个数\(-\)一个没选的生成树个数\(+\)两个没选的生成树个数\(-...\) 至于生成树个数怎么算,用\(Matrix - Tree\)矩阵树定理做就好了 如果不会:传送门 \(Code\) #include<bits/stdc++.h> #define int long long #defin