DP大作战—状态压缩dp

时间:2022-01-08 22:45:16

题目描述

阿姆斯特朗回旋加速式阿姆斯特朗炮是一种非常厉害的武器,这种武器可以毁灭自身同行同列两个单位范围内的所有其他单位(其实就是十字型),听起来比红警里面的法国巨炮可是厉害多了。现在,零崎要在地图上布置一片阿姆斯特朗回旋加速式阿姆斯特朗炮,那么在N行M列单位长度大小的地图上,求解阿姆斯特朗回旋加速式阿姆斯特朗炮最大的部署数量和对应部署方案总数。

输入

每组输入一行,为两个整数N,M (N <= 100;M <= 10)

输出

每组一行两个整数,

第一个为阿姆斯特朗回旋加速式阿姆斯特朗炮的个数,第二个为此数量下的摆放方式总数。

输入样例

3 3

输出样例

3 6

Hint

100   1为阿姆斯特朗回旋加速式阿姆斯特朗炮的位置,方案不唯一。
010
001

状态压缩是一种特殊的技巧,不止可以用在dp中。状态压缩其实是在利用数据结构,由于基础数据类型int有32位,一个int变量就可以表示2^32个状态。状态压缩其实是一种优化方法,有很大局限性。第一,单元状态通常只有两种(0、1,其实多了压缩的道理是一样的,但是没有位运算就没什么优势了),第二,单元维度通常不超过50 ( int才32,多了longlong都爆掉难道用高精度大整数压缩不成……)

状压dp本质上还是dp,问题大多数还是求解最大值或者解决方案总数之类的,但是状态数量巨大普通方法难以表示,所以利用整数可以将状态维度压缩到1维。

题目来源:http://biancheng.love/contest/10/problem/G/index

解题思路:

状态压缩dp问题。

关于状态压缩问题参见:http://www.cnblogs.com/avril/p/3282295.html

其他状态压缩DP问题:【POJ3254】【POJ1185】【POJ3311】【HDU3001】【POJ2288】【ZOJ4257】【POJ2411】【HDU3681】

给出本题代码:

 #include <bits/stdc++.h>
#define INF 99999999
typedef long long LL;
using namespace std; const int MAX=+;
int n,m,lastsize,lastlastsize,nowsize;
int last[MAX],lastlast[MAX],now[MAX];
int num[MAX],dp[MAX][MAX],temp[MAX][MAX];//dp[k][i][j]表示第k行选择i方案,第k-1行选择j方案的最大炮兵数 void dfs(int id,int k,int p,int sum)
{
if(k>=m)
{
now[++nowsize]=p;
num[nowsize]=sum;
return;
}
dfs(id,k+,p|(<<k),sum+);
dfs(id,k+,p,sum);
} void DP()
{
for(int k=; k<=n; ++k)
{
memset(now,,sizeof now);
nowsize=;
dfs(k,,,);
for(int i=; i<=nowsize; ++i)for(int j=; j<=lastsize; ++j)dp[i][j]=;
for(int i=; i<=nowsize; ++i) //本行选择第几个方案
{
for(int j=; j<=lastsize; ++j) //上一行选择第几个方案
{
for(int t=; t<=lastlastsize; ++t) //上上行选择第几个方案
{
if(now[i] & last[j])continue;//与上一行j方案不能共存
if(now[i] & lastlast[t])continue;//与上上行t方案不能共存
if(dp[i][j]<temp[j][t]+num[i])dp[i][j]=temp[j][t]+num[i];
}
}
}
for(int i=; i<=nowsize; ++i)for(int j=; j<=lastsize; ++j)temp[i][j]=dp[i][j];
for(int i=; i<=lastsize; ++i)lastlast[i]=last[i];
lastlastsize=lastsize;
for(int i=; i<=nowsize; ++i)last[i]=now[i];
lastsize=nowsize;
}
} int main()
{
while(~scanf("%d%d",&n,&m))
{
last[]=lastlast[]=temp[][]=;
lastsize=lastlastsize=;
DP();
int sum=,cot;
for(int i=; i<=lastsize; ++i)
{
for(int j=; j<=lastlastsize; ++j)
{
if(temp[i][j]>sum)
{
sum=temp[i][j];
cot=;
}
else if(temp[i][j]==sum)
cot++;
}
}
printf("%d %d\n",sum,cot);
}
return ;
}