●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;
}
●BZOJ 4596 [Shoi2016]黑暗前的幻想乡的更多相关文章
- bzoj 4596 [Shoi2016]黑暗前的幻想乡 矩阵树定理+容斥
4596: [Shoi2016]黑暗前的幻想乡 Time Limit: 20 Sec Memory Limit: 256 MBSubmit: 559 Solved: 325[Submit][Sta ...
- BZOJ 4596: [Shoi2016]黑暗前的幻想乡
Sol 容斥原理+Matrix-Tree定理.容斥跟小星星那道题是一样的,然后...直接Matrix-Tree定理就可以了... 复杂度\(O(2^{n-1}n^3)\) PS:调了好久啊QAQ 明明 ...
- bzoj 4596: [Shoi2016]黑暗前的幻想乡【容斥原理+矩阵树定理】
真是简单粗暴 把矩阵树定理的运算当成黑箱好了反正我不会 这样我们就可以在O(n^3)的时间内算出一个无向图的生成树个数了 然后题目要求每个工程队选一条路,这里可以考虑容斥原理:全选的方案数-不选工程队 ...
- BZOJ 4596: [Shoi2016]黑暗前的幻想乡(容斥+Matrix_Tree)
传送门 解题思路 看到计数想容斥--\(from\) \(shadowice1984\)大爷.首先求出原图的生成树个数比较容易,直接上矩阵树定理,但这样会多算一点东西,会把\(n-2\)个公司的多算进 ...
- 【BZOJ 4596】 4596: [Shoi2016]黑暗前的幻想乡 (容斥原理+矩阵树定理)
4596: [Shoi2016]黑暗前的幻想乡 Time Limit: 20 Sec Memory Limit: 256 MBSubmit: 324 Solved: 187 Description ...
- 【BZOJ】4596: [Shoi2016]黑暗前的幻想乡
[题意]给定n个点的无向完全图,有n-1个公司各自分管一部分路,要求所有公司都有修路的生成树数.n<=17. [算法]容斥原理+生成树计数(矩阵树定理) [题解]每个生成树方案是一个公司有无修路 ...
- bzoj4596[Shoi2016]黑暗前的幻想乡 Matrix定理+容斥原理
4596: [Shoi2016]黑暗前的幻想乡 Time Limit: 20 Sec Memory Limit: 256 MBSubmit: 464 Solved: 264[Submit][Sta ...
- bzoj4596/luoguP4336 [SHOI2016]黑暗前的幻想乡(矩阵树定理,容斥)
bzoj4596/luoguP4336 [SHOI2016]黑暗前的幻想乡(矩阵树定理,容斥) bzoj Luogu 题解时间 看一看数据范围,求生成树个数毫无疑问直接上矩阵树定理. 但是要求每条边都 ...
- [ZJOI2016]小星星&[SHOI2016]黑暗前的幻想乡(容斥)
这两道题思路比较像,所以把他们放到一块. [ZJOI2016]小星星 题目描述 小Y是一个心灵手巧的女孩子,她喜欢手工制作一些小饰品.她有n颗小星星,用m条彩色的细线串了起来,每条细线连着两颗小星星. ...
随机推荐
- 网络1712--c语言嵌套循环作业总结
1.助教有话说 首先,每周一篇的博客作业是很有必要的:编程的过程不仅仅是会敲几行代码.能够通过PTA就大吉大利了,你更应该做到的是梳理代码思路,通过与他人代码思路的比对,取其精华,进而不断进阶,才能逐 ...
- Alpha冲刺Day10
Alpha冲刺Day10 一:站立式会议 今日安排: 由林静完成第三方机构的用户信息管理模块 由张梨贤完成第三方机构的委托授权管理模块 由黄腾飞和周静平完成政府人员模块下风险管控子模块下的核实企业风险 ...
- SQLAlchemy 教程 —— 基础入门篇
SQLAlchemy 教程 -- 基础入门篇 一.课程简介 1.1 实验内容 本课程带领大家使用 SQLAlchemy 连接 MySQL 数据库,创建一个博客应用所需要的数据表,并介绍了使用 SQLA ...
- Android广播发送失败
现在至今为止Android 8.0 不支持大部分广播收发 如果无法使用建议换至Android 7.0版本 且 minSdkVersion 24
- 【Fungus入门】10分钟快速构建Unity中的万能对话系统 / 叙事系统 / 剧情系统
我真的很久没有写过一个完整的攻略了(笑),咸鱼了很久之后还是想来写一个好玩的.这次主要是梳理一下Unity的小众插件Fungus的核心功能,并且快速掌握其使用方法. 官方文档:http://fungu ...
- JAVA_SE基础——36.static的实际应用
什么时候定义静态函数 如果功能内部没有访问到非静态数据(对象的特有数据.那么该功能就可以定义为静态) P.S. 静态方法作为类和接口的重要组成部分,可以通过类名或接口直接访问,通常将那些使用频率较高的 ...
- django models的点查询/跨表查询/双下划线查询
django models 在日常的编程中,我们需要建立数据库模型 而往往会用到表与表之间的关系,这就比单表取数据要复杂一些 在多表之间发生关系的情形下,我们如何利用models提供的API的特性获得 ...
- Mego开发文档 - 数据库建模
数据库建模 我们还提供了一些其他的特性,用于定制化数据库对应的数据结构. 表映射 框架默认会使用CLR类型名称做为实际数据库的表名,当两者不一致时可以使用该特性强制表名称. [Table(" ...
- python之celery的使用(一)
前段时间需要使用rabbitmq做写缓存,一直使用pika+rabbitmq的组合,pika这个模块虽然可以很直观地操作rabbitmq,但是官方给的例子太简单,对其底层原理了解又不是很深,遇到很多坑 ...
- 新概念英语(1-115)Knock! Knock!
Lesson 115 Knock, knock! 敲敲门! Listen to the tape then answer this question. What does Jim have to dr ...