[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++

原文地址:https://www.cnblogs.com/wzc521/p/11619687.html

时间: 2024-11-10 13:06:47

[CSP-S模拟测试]:巨神兵(状压DP)的相关文章

UVA 1412 - Fund Management(用vector容器模拟状态的状压dp)

Frank is a portfolio manager of a closed-end fund for Advanced Commercial Markets (ACM ). Fund collects money (cash) from individual investors for a certain period of time and invests cash into various securities in accordance with fund's investment

NOJ 1116 哈罗哈的大披萨 【淡蓝】 [状压dp+各种优化]

我只能说,珍爱生命,远离卡常数的题...感谢陈老师和蔡神,没有他们,,,我调一个星期都弄不出来,,,, 哈罗哈的大披萨 [淡蓝] 时间限制(普通/Java) : 1000 MS/ 3000 MS          运行内存限制 : 65536 KByte总提交 : 73            测试通过 : 9 描述 热风哈罗哈(三牌楼)店正在搞活动:他们将提供一个大披萨给第一个告诉他们以下信息的人:一次购买任一种披萨,哪种单位面积的价格最低.问题初步想一想,好像是这么解决:“对于每个披萨计算平均

(九度OJ)题目1338:角斗士(状压DP)

题目描述: 角斗士是古罗马奴隶社会的一种特殊身份的奴隶,他们的职责是在角斗场上进行殊死搏斗,为了人们提供野蛮的娱乐.他们的结局或是战死,或者由于表现突出赢得胜利而获得释放. 现在在角斗场里有N个待战的角斗士(1 <=N<=18),每天都将举行一场比赛,为了保持比赛的刺激性,每场比赛前才会在所有当前活着的角斗士之中随机选择两名进行上场拼杀.每场比赛的结束条件即为其中一名被杀死.当进行了N场比赛之后,最后存活的角斗士将被释放.而你将被赋予一个任务,计算出每名角斗士最终存活的概率.我们将提供角斗士之

HDU 1565&amp;1569 方格取数系列(状压DP或者最大流)

方格取数(2) Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) Total Submission(s): 6206    Accepted Submission(s): 1975 Problem Description 给你一个m*n的格子的棋盘,每个格子里面有一个非负数. 从中取出若干个数,使得任意的两个数所在的格子没有公共边,就是说所取数所在的2个格子不能相邻,并且取出的

UVa 11825 (状压DP) Hackers&#39; Crackdown

这是我做状压DP的第一道题,状压里面都是用位运算来完成的,只要耐下心来弄明白每次位运算的含义,还是容易理解的. 题意: 有编号为0~n-1的n台服务器,每台都运行着n中服务,每台服务器还和若干台其他服务器相连.对于每台服务器,你可以选择停止该台以及与这台服务器相连的服务器的一项服务.如果一台服务器的所有服务都被停止,则这台服务器瘫痪.问最多能使多少台服务器瘫痪 转化为数学模型(题目是如何抽象成这种数学模型的也要好好想想): 把n个集合尽可能多的分成若干组,使得每组所有集合的并集为全集.这里集合P

[POJ1038]状压DP

题意:给一个n*m的区域,里面有一些障碍物,往里面放2*3和3*2的矩形,矩形之间不能重叠,不能覆盖到障碍物,求能放置的最大个数.(n<=150,m<=10) 思路:看到m=10就应该往状压dp方面想了.由于有3*2的矩形,所以需要记录2行的状态,粗略估计状态数高达150*2^20=1.5*1e8,这么多状态必然超时,注意到如果(i-1,j)为0了,无论(i,j)为1或0,(i,j)都不能放矩形,于是知道有很多无用的或者说不合法的状态,两行的状态用m位3进制数表示同样能实现转移.由于3进制数操

【POJ 2411】Mondriaan&#39;s Dream(状压dp)

[POJ 2411]Mondriaan's Dream(状压dp) Time Limit: 3000MS   Memory Limit: 65536K Total Submissions: 14107   Accepted: 8152 Description Squares and rectangles fascinated the famous Dutch painter Piet Mondriaan. One night, after producing the drawings in hi

2015广东工业大学ACM校赛 I 游戏王 (状压dp)

Problem I: 游戏王 Description 小学的时候,Stubird非常喜欢玩游戏王,有一天,他发现了一个绝佳的连锁组合,这个连锁组合需要6张卡, 可是他一张都没有,但是他的那些朋友们有,不过当然,他们不会白给,不过也不排除有人和他交情好,送给他了. 不过他们有成全别人的美德,当他们看到Stubird已经有某些他们喜欢的卡的时候,他们会给他优惠,或者更贵也说不定 嘛不过,你可以把有的卡片藏起来,不告诉他们,来获得更低的价格. 问他最少需要多少钱才可以集齐所有的卡. Input 第一行

[DP总结]状压DP

顾名思义,是用将状态进行二进制压缩成集合的形式来方便DP转移的方法. 一些常用的代码表示如下 i & j //取状态i,j重合部分 i ^ j //取状态i,j不同部分 i | j //合并状态i,j (1 << N) - 1 //表示111-1(N个1) 1 << i - 1 //表示00100-0(1后面有i-1个0,也就是有且仅有二进制下第i位为1) for (int i = 0; i < n; ++ i) if (x & (1 << i))

v&#183;y「状压dp」

一直对状压dp怀有一种恐惧感 不会打,不会调,关键是不会调 做了这两道题,虽然还是不会状压dp,但总比之前好了一些 y 普通状压应该很好打 复杂度$O(2^d*n*(n+m))$ for(ll i=2;i<=d;i++){ for(ll state=0;state<=maxn;state++){ for(ll x=1;x<=n;x++){ for(ll j=head[x];j;j=nxt[j]){ ll y=ver[j]; if(!f[i-1][x][state]) continue ;