POJ1185 炮兵阵地 状态压缩
因为不知道不同的博客怎么转,就把别人的复制过来了,这个题解写的非常好,原地址为:
http://hi.baidu.com/wangxustf/item/9138f80ce2292b8903ce1bc7
分类:DP
题目分析与算法模型
因为行N <= 100 列M <= 10
我们就在10上面做文章。
在任意一行上,最多10列,假设这10个位置都是平原,那么在这10个位置上放置炮兵并互不攻击,一共有60种方法(可以算一下)。具体是哪60种方法,这个需要枚举一下。我们用num[i]表示第i种方案放置的炮兵数量。
比如:
PPPPHHPPPP
我们可以在第1个位置和第7个位置各放一个炮兵,这是一种可行的方案。此时num[i] = 2
假设一共有N行,
对于第s-2行,它有60种方案
对于第s-1行,它也有60种方案
对于第s行,它也有60种方案
对于第s+1行,它也有60种方案
.......
我们用dp[s][i][j]表示第s行使用第i种方案,第s-1行使用第j种方案时部队可以部署的最大值.那么max{dp[N][i][j], 其中i,j=1...60}的就是最后的答案
这里需要检验状态i和j是否互相兼容,这个可以通过连个状态的十进制表示的两个数间的&运算来判断(如果兼容,那么运算后的值必为0)
如果兼容,可以推出:
dp[s][i][j] = num[i] + max{dp[s-1][j][k], 其中k=1...60,并且i和k状态也时兼容的}
这表示:第s行使用第i种方案,第s-1行使用第j种方案时,我们枚举第s-2行的方案k。如果i,j,k这三种方案之间是相容的,那么dp[s][i][j] = num[i] + dp[s-1][j][k]
代码:
#include<stdio.h>
#include<string.h>
int cnt,m,n;
int dp[100][64][64],num[64],state[64],bitmap[100];
void init()
{
int tmp;
cnt=0;
for(int i=0;i<(1<<m);i++) //枚举每行的状态,从0到2^m - 1,判断其是否合法
{
tmp=i;
if( ((tmp<<1)&i) | ((tmp<<2)&i) ) continue;//判断该行在这个状态时是否合法(任意炮兵都不在其他炮兵的攻击范围之内)
state[cnt]=i; //通过数组state[]记录合法的状态(十进制表示)
num[cnt]=tmp&1; //num[]数组记录这个合法状态下‘1’的个数(也就时炮兵的个数)
while( tmp = (tmp>>1) )
num[cnt]+=tmp&1;
cnt++; //此函数的统计是假设当该行都为平地时,即共有cnt+1个合法状态
}
}
void solve()
{
int ans,i,j,k,p;
memset(dp,0,sizeof(dp));
for(i=0;i<n;i++) //枚举每一行
for(j=0;j<cnt;j++) //先枚举第i行的可能状态
{
if(bitmap[i]&state[j]) continue; //地图中标记为“山地”的点不能布兵
if(i==0) dp[i][j][0]=num[j];
else if(i==1)
{
for(k=0;k<cnt;k++) //枚举第i-1行的可能状态
{
if(bitmap[i-1]&state[k]) continue; //判断第i-1行的k状态是否和山地冲突,冲突就跳到下一个状态k+1
if(state[j]&state[k]) continue; //判断上下两行(i-1和i)的合法状态是否兼容彼此
if(dp[i][j][k]<dp[i-1][k][0]+num[j])
dp[i][j][k]=dp[i-1][k][0]+num[j];
}
}
else
{
for(k=0;k<cnt;k++) //枚举第i-1行的可能状态
{
if(bitmap[i-1]&state[k]) continue; //判断第i-1行的k状态是否和山地冲突,冲突就跳到下一个状态k+1
if(state[j]&state[k]) continue; //判断上下两行(i-1和i)的合法状态是否兼容彼此
for(p=0;p<cnt;p++) //枚举第i-2行的可能状态
{
if(bitmap[i-2]&state[p]) continue; //判断第i-2行的p状态是否和山地冲突,冲突就跳到下一个状态p+1
if(state[k]&state[p] || state[j]&state[p]) continue;//判断上下两行(i-2和i-1,i-2和i)的合法状态是否兼容彼此
if(dp[i][j][k]<dp[i-1][k][p]+num[j])
dp[i][j][k]=dp[i-1][k][p]+num[j];
}
}
}
}
ans=0;
//for(i=0;i<n;i++)
for(j=0;j<cnt;j++)
for(k=0;k<cnt;k++)
if(dp[n-1][j][k]>ans)
ans=dp[n-1][j][k];
printf("%d\n",ans);
}
int main()
{
char s[12];
int i,j;
while(scanf("%d%d",&n,&m)!=EOF)
{
memset(bitmap,0,sizeof(bitmap));
for(i=0;i<n;i++)
{
scanf("%s",s);
for(j=0;j<m;j++)if(s[j]=='H')bitmap[i]+=(1<<(m-1-j));
//if(s[j]=='H')bitmap[i]|=(1<<j);
}
init();
solve();
}
return 0;
}
/*算法回顾:
压缩状态的动态规划
1.利用二进制表示某个位置有无炮兵状态
2.把任意一行看成一个二进制串,并转化为相应的十进制数,即该行的压缩状态
3.枚举该行(i)所有二进制串,找出单独一行的所有可能的状态,即合法的压缩状态(s[i])
4.利用滚动数组降低内存消耗
技巧: 二进制的位运算,滚动数组
动规方程: f[i][j][k] = max{f[i-1][k][l]+c[j]},
f[i][j][k]表示第i行状态为s[j],第i-1行状态为s[k]的最大炮兵数
枚举l的每种状态,且s[j],s[k],s[l],地形互不冲突
*/
POJ1185 炮兵阵地 状态压缩的更多相关文章
- POJ1185 - 炮兵阵地(状态压缩DP)
题目大意 中文的..直接搬过来... 司令部的将军们打算在N*M的网格地图上部署他们的炮兵部队.一个N*M的地图由N行M列组成,地图的每一格可能是山地(用"H" 表示),也可能是平 ...
- POJ1185炮兵阵地(状态压缩 + dp)
题目链接 题意:给出一张n * m的地图,其中 有的地方能放大炮,有的地方不能,大炮与上下左右两个单位范围内会相互攻击,问最多能放几个大炮 能放大炮为1不能放大炮为0,把每一行看做一个状态,要除去同一 ...
- POJ - 1185 炮兵阵地 (状态压缩)
题目大意:中文题目就不多说大意了 解题思路: 1.每行最多仅仅有十个位置,且不是山地就是平原,那么就能够用1表示山地,0表示平原,将每一行的状态进行压缩了 2.接着找出每行能放炮兵的状态.先不考虑其它 ...
- [P2704][NOI2001]炮兵阵地 (状态压缩)
最近抄状压的代码…… 然后盯上了这个题目 调试了一个晚上,终于A了 但是是对着宝典打的,我依然不懂状态压缩 那么下一步先把装压放一放,学一下树形DP吧 #include<cstdio> # ...
- luogu2704 炮兵阵地 状态压缩DP
题目大意:一个N*M的地图由N行M列组成,地图的每一格可能是山地(用“H” 表示),也可能是平原(用“P”表示),在每一格平原地形上最多可以布置一支炮兵部队,能攻击到的区域:沿横向左右各两格,沿纵向上 ...
- poj 1185 炮兵阵地 状态压缩dp
思路:定义一个三维数组dp[x][i][j]其中x为now和pre两种状态,now表示当前两行最优解,pre表示出了本行外,前两行的最优解.那么状态转移方程为 dp[now][j][k]=max(dp ...
- POJ 3254 炮兵阵地(状态压缩DP)
题意:由方格组成的矩阵,每个方格可以放大炮用P表示,不可以放大炮用H表示,求放最多的大炮,大炮与大炮间不会互相攻击.大炮的攻击范围为两个方格. 分析:这次当前行的状态不仅和上一行有关,还和上上行有关, ...
- POJ1185炮兵阵地(状态压缩DP)
POJ飞翔.数据弱 ZQOJ飞翔 数据强 Description 司令部的将军们打算在N×M的网格地图上部署他们的炮兵部队.一个N×M的地图由N行M列组成,地图的每一格可能是山地(用"H&q ...
- [poj1185]炮兵阵地_状压dp
炮兵阵地 poj-1185 题目大意:给出n列m行,在其中添加炮兵,问最多能加的炮兵数. 注释:n<=100,m<=10.然后只能在平原的地方建立炮兵. 想法:第2到状压dp,++.这题显 ...
随机推荐
- windows下安装NodeJs
1.官网(//nodejs.org/en/)下载系统匹配的文件 2.双击安装,完成后发现nodejs文件夹下面有npm, 直接用npm安装其他环境既可 3.如果配置了环境变量,直接Win+R后CMD调 ...
- android玩耍(-) adbshell安装
一 什么是adbshell http://adbshell.com/ Android Debug Bridge (adb) is a command line tool that lets you c ...
- td太多内容显示...
table style="table-layout:fixed;"td style="text-overflow: ellipsis;white-space: nowra ...
- [原创]NT系统信息察看工具 : NtInfoGuy
原文链接:[原创]NT系统信息察看工具 : NtInfoGuy 对于windows的内部,我们有太多的东西需要了解,认知.我们非凡的.从不知足的探求本性驱使我们要 拨开迷雾得见青天.太多的木马,病毒, ...
- 论山寨手机与Android联姻 【10】SmartPhone的通信机制
上一章我们说到,智能手机 == 电脑 + 移动网卡,这个提法比较粗略,更精准的提法应当是,智能手机的硬件结构分为应用程序处理器AP,和基带处理器BP两个部分.虽然AP部分的功能与电脑主板基本类似,但是 ...
- 进程占用百分百CPU不卡(从未试过,当别的程序运行的时候,当前程序还会运行吗?)
在写程序中.为了让程序效率高.有时会点用很高的CPU.这里用户体验不好可以设置线程的优先级来搞定. BOOL SetThreadPriority( HANDLE hThread, // handle ...
- Flex中如何通过horizontalTickAligned和verticalTickAligned样式指定线图LineChart横竖方向轴心标记的例子
原文http://blog.minidx.com/2008/12/03/1669.html 接下来的例子演示了Flex中如何通过horizontalTickAligned和verticalTickAl ...
- GDOI模拟赛Round 1
GDOI模拟赛Round 1 数据结构 题目描述:给出一个长度为\(n\)的序列,支持两种操作: 1.对某段区间都加上一个数 2.给出\(p.k\),求下面表示式对\((10^9+7)\)取模 \[\ ...
- CSS滤镜让图片模糊(毛玻璃效果)实例页面
<pre name="code" class="css">CSS代码: .blur { filter: url(blur.svg#blur); /* ...
- 04737_C++程序设计_第1章_认识C++的对象
例1.1 演示使用结构对象的示例程序. //功能:将结构对象的两个域值相加,乘以2再加50 #include <iostream>//包含头文件 using namespace std;/ ...