[CSP-S模拟测试]:巨神兵(状压DP)
题目描述
欧贝利斯克的巨神兵很喜欢有向图,有一天他找到了一张$n$个点$m$条边的有向图。
欧贝利斯克认为一个没有环的有向图是优美的,请问这张图有多少个子图(即选定一个边集)是优美的?答案对$1,000,000,007$取模。
输入格式
第一行两个整数$n$和$m$。
接下来$m$行每行两个整数表示一条有向边。保证无重边无自环。
输出格式
一行一个整数表示答案,对$1,000,000,007$取模。
样例
样例输入:
3 6
1 2
2 1
1 3
3 1
2 3
3 2
样例输出:
25
数据范围与提示
对于$40\%$的数据$n\leqslant 5,m\leqslant 20$;
对于$60\%$的数据$n\leqslant 10$;
对于$80\%$的数据$n\leqslant 15$;
对于$100\%$的数据$n\leqslant 17$。
题解
看到数据范围是$17$的时候,我觉得这场考试我能$AK$;然而,当我看到是边集的时候,我发现半个小时之后机房里听不见一点键盘声……
同样,我们先考虑$60\%$的做法。
先给有向无环图分层,第一层是入度为$0$的点,第二层是删去第一层后度数为$0$的点,类似$topsort$?
然后设$dp[i][j]$表示当前选择的节点集合为$i$,最后一层的节点集合为$j$的方案数。
不妨设$s_1$表示当前选择的节点集合,$s_2$表示最后一层的节点的集合,$s_3$为$s_1$补集的子集,且$s_2$与$s_3$有连边,$s_1\oplus s_2$与$s_3$的连边为$cnt_1$条,$s_2$与$s_3$的连边有$cnt_2$条。
那么,状态转移方程为:
$$dp[s_1|s_3][s_3]=\sum dp[s_1][s_2]\times \prod 2^{cnt_1}\times (2^{cnt_2}-1)$$
这样的时间复杂度是$\Theta(4^n\times m)$的,无论空间还是时间都是不行的。
所以我们考虑减掉第二维,直接枚举当前选择的节点集合$s_1$,然后再枚举其补集的子集$s_2$,设$s_1$与$s_2$之间的连边有$cnt$条。
那么,你可能会列出这样一个状态转移方程:$dp[s_1|s_2]=dp[s_1]\times 2^{cnt}$。
但是你会发现答案会统计多,因为$s_1|s_2$可以由很多$s_1$和$s_2$组成,所以我们还需要容斥,其系数为$(-1)^{sz[s]+1}$。
这时候的时间复杂度为$\Theta(3^nm)$,只能拿到$80$分,还需要优化,有好多优化,我将其优化到了$\Theta(2^n\times n^2)$,这样就可以$AC$这道题了。
时间复杂度:$\Theta(2^n\times n^2)$。
期望得分:$100$分。
实际得分:$100$分。
代码时刻
#include<bits/stdc++.h>
using namespace std;
const int mod=1000000007;
int n,m;
bool Map[20][20];
int sz[131073];
int du[65537];
int sum[131073];
int qpow[1000];
int dp[131073];
int main()
{
scanf("%d%d",&n,&m);
int maxn=(1<<n);
dp[0]=qpow[0]=1;
for(int i=1;i<=m;i++)
{
int x,y;
scanf("%d%d",&x,&y);
Map[x-1][y-1]=1;
qpow[i]=(qpow[i-1]<<1)%mod;
}
sz[0]=-1;
for(int i=1;i<maxn;i++)sz[i]=sz[i>>1]*(i&1?-1:1);
for(int i=0;i<maxn-1;i++)
{
if(!dp[i])continue;
int t=maxn-1-i;
memset(du,0,sizeof(du));
for(int j=0;j<n;j++)
if(i&(1<<j))
for(int k=0;k<n;k++)
du[1<<k]+=Map[j][k];
sum[0]=0;
for(int s=(t-1)&t;;s=(s-1)&t)
{
int now=t^s,lst=now&-now;
sum[now]=sum[now-lst]+du[lst];
dp[i+now]=(dp[i+now]+1LL*sz[now]*qpow[sum[now]]*dp[i])%mod;
if(!s)break;
}
}
printf("%lld",dp[maxn-1]);
return 0;
}
rp++
[CSP-S模拟测试]:巨神兵(状压DP)的更多相关文章
- NOIp模拟赛 巨神兵(状压DP 容斥)
\(Description\) 给定\(n\)个点\(m\)条边的有向图,求有多少个边集的子集,构成的图没有环. \(n\leq17\). \(Solution\) 问题也等价于,用不同的边集构造DA ...
- 2018.10.17 NOIP模拟 管道(状压dp)
传送门 状压dp好题. 怎么今天道道题都有点东西啊 对于今天题目神仙出题人先膜为上策:%%%%DzYoAk_UoI%%%% 设f[i][j]f[i][j]f[i][j]表示选取点的状态集合为iii,当 ...
- 2018.09.08 NOIP模拟 division(状压dp)
这么sb的题考场居然写挂了2233. 假设n=∏iaiki" role="presentation" style="position: relative;&qu ...
- 2018.08.29 NOIP模拟 movie(状压dp/随机化贪心)
[描述] 小石头喜欢看电影,选择有 N 部电影可供选择,每一部电影会在一天的不同时段播 放.他希望连续看 L 分钟的电影.因为电影院是他家开的,所以他可以在一部电影播放过程中任何时间进入或退出,当然他 ...
- 4.26 省选模拟赛 T3 状压dp 差分求答案
LINK:T3 比较好的题目 考试的时候被毒瘤的T2给搞的心态爆炸 这道题连正解的思路都没有想到. 一看到题求删除点的最少个 可以使得不连通. 瞬间想到最小割 发现对于10分直接跑最小割即可. 不过想 ...
- 【CSP模拟赛】Adore(状压dp 二进制)
题目描述 小w偶然间见到了一个DAG.这个DAG有m层,第一层只有一个源点,最后一层只有一个汇点,剩下的每一层都有k个节点.现在小w每次可以取反第i(1<i<n-1)层和第i+1层之间的连 ...
- [CSP-S模拟测试]:装饰(状压DP)
题目传送门(内部题114) 输入格式 第一行一个正整数$n$. 接下来一行$n-1$个正整数,第$i$个数为$f_{i+1}$. 接下来一行$n$个数,若第$i$个数为$0$则表示林先森希望$i$号点 ...
- 6.28 NOI模拟赛 好题 状压dp 随机化
算是一道比较新颖的题目 尽管好像是两年前的省选模拟赛题目.. 对于20%的分数 可以进行爆搜,对于另外20%的数据 因为k很小所以考虑上状压dp. 观察最后答案是一个连通块 从而可以发现这个连通块必然 ...
- 【62测试】【状压dp】【dfs序】【线段树】
第一题: 给出一个长度不超过100只包含'B'和'R'的字符串,将其无限重复下去. 比如,BBRB则会形成 BBRBBBRBBBRB 现在给出一个区间[l,r]询问该区间内有多少个字符'B'(区间下标 ...
- 2018.10.05 NOIP模拟 上升序列(状压dp)
传送门 状压dp好题. 首先需要回忆O(nlogn)O(nlog n)O(nlogn)求lislislis的方法,我们会维护一个单调递增的ddd数组. 可以设计状态f(s1,s2)f(s1,s2)f( ...
随机推荐
- Linux 下在后台运行进程:nohup,setsid,& 以及 tmux
参考: Linux 技巧:让进程在后台可靠运行的几种方法 ssh 登录了远程服务器时,如果在前台运行耗时较长的任务, 当 ssh 掉线或关闭窗口时会导致命令停止运行. hup 与 nohup 当用户注 ...
- PYTHON2.7之前需要独立安装pip
如果python2版本是>=2.7.9, python3版本是>=3.4, pip已将一起随python安装成功了. 对于Python 2.6,你需要更旧setuptools.适用于Pyt ...
- oracle--二维表的操作创建修改删除
oracle学习内容 oracle的管理系统学习 oracle的数据管理学习 oracle的用户管理 oracle二维表管理 创建表和字段讲解 --创建表学习 1. 创建表的基本语句:create t ...
- [Web 前端] 015 css 三种元素的介绍
1. 块元素,内联元素,内联块元素 元素就是标签 布局中常用的有三种标签 块元素 内联元素 内联块元素 1.1 块元素 也称为行元素 布局中常用的标签,如 div.p.ul.li.h1~h6.dl.d ...
- 从SVN下检出项目内容【步骤】
1.新创建一个新的工作环境,然后new--->other--->SVN 2.点击Next,然后进行检出项目的操作,如下图所示: 3.再点击Next,进行输入指定的url地址,从指定的url ...
- UVA 10003 Cutting Sticks 区间DP+记忆化搜索
UVA 10003 Cutting Sticks+区间DP 纵有疾风起 题目大意 有一个长为L的木棍,木棍中间有n个切点.每次切割的费用为当前木棍的长度.求切割木棍的最小费用 输入输出 第一行是木棍的 ...
- 洛谷 P1462 通往奥格瑞玛的道路(二分答案,堆优化dijkstra)
传送门 解题思路 首先看题目问题,求经过的所有城市中最多的一次收取的费用的最小值是多少.一看“最大值最小”就想到了二分答案. 在读一遍题目,就是二分收取的费用,然后对于每一个二分的费用,跑一边最短路, ...
- [BZOJ4476] [JSOI2015] 送礼物 (01分数规划+ST表)
[BZOJ4476] [JSOI2015] 送礼物 (01分数规划+ST表) 题面 给出n,k,l,r和序列a,要求从a中选一段连续的区间[i,j]出来,使得M(i,j)-m(i,j)/(j-i+k) ...
- 让网站动起来!12款优秀的 jQuery 动画
Textillate.js 介绍:Textillate.js 是一个简单的 CSS3 文本动画插件.结合了一些非常棒的库,把 CSS3 动画轻松应用到任何文本.只需要在项目中简单地引入 textill ...
- Vue 实现文件的上传
要把文件上传的web,需要分几步? 答:三步 第一步:创建一个上传文件的标签 <input type="file" id="fileExport" @ch ...