POJ 1185 炮兵阵地 (状态压缩DP)
Description
司令部的将军们打算在NM的网格地图上部署他们的炮兵部队。一个NM的地图由N行M列组成,地图的每一格可能是山地(用"H" 表示),也可能是平原(用"P"表示),如下图。在每一格平原地形上最多可以布置一支炮兵部队(山地上不能够部署炮兵部队);一支炮兵部队在地图上的攻击范围如图中黑色区域所示:

如果在地图中的灰色所标识的平原上部署一支炮兵部队,则图中的黑色的网格表示它能够攻击到的区域:沿横向左右各两格,沿纵向上下各两格。图上其它白色网格均攻击不到。从图上可见炮兵的攻击范围不受地形的影响。
现在,将军们规划如何部署炮兵部队,在防止误伤的前提下(保证任何两支炮兵部队之间不能互相攻击,即任何一支炮兵部队都不在其他支炮兵部队的攻击范围内),在整个地图区域内最多能够摆放多少我军的炮兵部队。
Input
第一行包含两个由空格分割开的正整数,分别表示N和M;
接下来的N行,每一行含有连续的M个字符('P'或者'H'),中间没有空格。按顺序表示地图中每一行的数据。N <= 100;M <= 10。
Output
仅一行,包含一个整数K,表示最多能摆放的炮兵部队的数量。
Sample Input
5 4
PHPP
PPHH
PPPP
PHPP
PHHP
Sample Output
6
分析:
实现需要考虑到一点就是,我们在进行放炮兵的时候,肯定是有一定的方向性的,所以在判断当前这个点能不能放置炮兵的时候,我们只需要考虑它的上一行和上上一行炮兵放置的状态就可以了。
当然还有一点就是如果当前这个点放置了炮兵的话,为了防止本行内的炮兵之间相互攻击,那么下一个需要判断的能不能放置炮兵的地方就是它本身往右走第三个位置,可以直接跳过去;如果当前这个位置没有放置炮兵的话,就得一个位置一个位置的挨着进行判断。
用0和1来表示一个位置有没有放炮兵,1表示放了,0表示没有,这样每一行的放置炮兵的状态都可以用二进制数来表示,二进制数之间进行与运算,就可以判断出来这一行的当前位置如果放置炮兵的话会不会与之前的情况产生冲突。
代码:
#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <string>
#include <queue>
#include <algorithm>
#include <map>
#include <cmath>
#include <iomanip>
#define INF 99999999
typedef long long LL;
using namespace std;
const int MAX=100+10;
int n,m,lastsize,lastlastsize,nowsize; //nowsize当前行的方案数,lastsize上一行的方案数,lastlastsize上上一行的方案数
int last[MAX],lastlast[MAX],now[MAX]; // now[i]第i行的方案......
int num[MAX],dp[MAX][MAX],temp[MAX][MAX]; //dp[i][j] i代表当前行 j代表上一行
//temp[j][t] j上一行 t上上行
char Map[MAX][MAX];
void dfs(int id,int k,int p,int sum)
{
if(k>=m)
{
now[++nowsize]=p;
num[nowsize]=sum;
return;
}
if(Map[id][k] == 'P') //找到可安放炮兵的位置
dfs(id,k+3,p|(1<<k),sum+1);
dfs(id,k+1,p,sum);
}
void DP()
{
for(int k=1; k<=n; ++k)
{
memset(now,0,sizeof now);
nowsize=0;
dfs(k,0,0,0);
for(int i=1; i<=nowsize; ++i)
for(int j=1; j<=lastsize; ++j)
dp[i][j]=0;
for(int i=1; i<=nowsize; ++i) //本行选择第几个方案
{
for(int j=1; j<=lastsize; ++j) //上一行选择第几个方案
{
for(int t=1; 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=1; i<=nowsize; ++i) //将当前状态赋给上一次的, 进行下次操作
for(int j=1; j<=lastsize; ++j)
temp[i][j]=dp[i][j];
for(int i=1; i<=lastsize; ++i)
lastlast[i]=last[i];
lastlastsize=lastsize;
for(int i=1; i<=nowsize; ++i)
last[i]=now[i];
lastsize=nowsize;
}
}
int main()
{
while(~scanf("%d%d",&n,&m))
{
for(int i=1; i<=n; ++i)scanf("%s",Map[i]);
last[1]=lastlast[1]=temp[1][1]=0;
lastsize=lastlastsize=1;
DP();
int sum=0;
for(int i=1; i<=lastsize; ++i) //遍历temp(也就相当于是dp)值 找到其中最大的sum值
{
for(int j=1; j<=lastlastsize; ++j)
{
if(temp[i][j]>sum)
sum=temp[i][j];
}
}
printf("%d\n",sum);
}
return 0;
}
POJ 1185 炮兵阵地 (状态压缩DP)的更多相关文章
- poj 1185 炮兵阵地 状态压缩dp
思路:定义一个三维数组dp[x][i][j]其中x为now和pre两种状态,now表示当前两行最优解,pre表示出了本行外,前两行的最优解.那么状态转移方程为 dp[now][j][k]=max(dp ...
- POJ - 1185 炮兵阵地 (状态压缩)
题目大意:中文题目就不多说大意了 解题思路: 1.每行最多仅仅有十个位置,且不是山地就是平原,那么就能够用1表示山地,0表示平原,将每一行的状态进行压缩了 2.接着找出每行能放炮兵的状态.先不考虑其它 ...
- POJ 3254 炮兵阵地(状态压缩DP)
题意:由方格组成的矩阵,每个方格可以放大炮用P表示,不可以放大炮用H表示,求放最多的大炮,大炮与大炮间不会互相攻击.大炮的攻击范围为两个方格. 分析:这次当前行的状态不仅和上一行有关,还和上上行有关, ...
- POJ 1185 炮兵阵地 状压dp
题目链接: http://poj.org/problem?id=1185 炮兵阵地 Time Limit: 2000MS Memory Limit: 65536K 问题描述 司令部的将军们打算在N*M ...
- POJ1185 - 炮兵阵地(状态压缩DP)
题目大意 中文的..直接搬过来... 司令部的将军们打算在N*M的网格地图上部署他们的炮兵部队.一个N*M的地图由N行M列组成,地图的每一格可能是山地(用"H" 表示),也可能是平 ...
- poj - 1185 炮兵阵地 状压DP 解题报告
炮兵阵地 Time Limit: 2000MS Memory Limit: 65536K Total Submissions: 21553 Accepted: 8363 Description ...
- POJ 1185炮兵阵地 (状压DP)
题目链接 POJ 1185 今天艾教留了一大堆线段树,表示做不动了,就补补前面的题.QAQ 这个题,我第一次写还是像前面HDU 2167那样写,发现这次影响第 i 行的还用i-2行那样,那以前的方法就 ...
- [poj 1185] 炮兵阵地 状压dp 位运算
Description 司令部的将军们打算在N*M的网格地图上部署他们的炮兵部队.一个N*M的地图由N行M列组成,地图的每一格可能是山地(用"H" 表示),也可能是平原(用&quo ...
- luogu2704 炮兵阵地 状态压缩DP
题目大意:一个N*M的地图由N行M列组成,地图的每一格可能是山地(用“H” 表示),也可能是平原(用“P”表示),在每一格平原地形上最多可以布置一支炮兵部队,能攻击到的区域:沿横向左右各两格,沿纵向上 ...
- POJ1185炮兵阵地(状态压缩 + dp)
题目链接 题意:给出一张n * m的地图,其中 有的地方能放大炮,有的地方不能,大炮与上下左右两个单位范围内会相互攻击,问最多能放几个大炮 能放大炮为1不能放大炮为0,把每一行看做一个状态,要除去同一 ...
随机推荐
- UDJC用户自定义Java类
private RowSet t1 = null;//业务表步骤 private RowSet t2 = null;//删除步骤 public boolean processRow(StepMetaI ...
- shell 指令 摘录
作者: learner811 本文摘自网络 权限:用户 读写 ugoa rwx * useradd userdel groupadd groupmod usermod su sudo wh ...
- CodeForces - 988D(思维STL)
原文地址:https://blog.csdn.net/weixin_39453270/article/details/80548442 博主已经讲的很好了 题意: 从一个序列中,选出一个集合,使得集合 ...
- What Kind of Friends Are You? ZOJ - 3960(ZheJiang Province Contest)
怎么说呢...我能说我又过了一道水题? emm... 问题描述: 给定 n 个待确定名字的 Friends 和 q 个问题.已知 c 个 Friends 的名字. 对于第 i 个问题,有 个 Fri ...
- (待修莫队 没过! 抽空在检查)Dynamic len(set(a[L:R])) UVA - 12345
#include <iostream> #include <cstdio> #include <sstream> #include <cstring> ...
- IbatisNet连接oracle 报错
提示什么 connect oracle 1.5.0.xxxx 将你本机的oracle 客户端版本重装换成32位即可
- vs2013 查找进行的过程中被停止
VS"Find in Files"失效的解决方法一:让VS窗口获得焦点,依次按以下快捷键Ctrl+BreakCtrl+Scroll LockAlt+Break VS"Fi ...
- php将两张身份证图片合并到一张图
/** * @desc 合并身份证的正反面到同一张图片 * @author Jimmy * @date 2016-12-33 * @param $imageSrc0 身份证正面 * @param $i ...
- AptanaStudio3+PHP程序远程调试的方法和步骤
php是执行在服务器上的脚本程序,通常调试bug,直接在浏览器页面就可以打印出错误信息,凭此基本能解决所有bug,但是有时候,可以说大多数时候,php会直接处理客户端的请求,作为数据接口传递数据,没有 ...
- Luogu 3810 & BZOJ 3262 陌上花开/三维偏序 | CDQ分治
Luogu 3810 & BZOJ 3263 陌上花开/三维偏序 | CDQ分治 题面 \(n\)个元素,每个元素有三个值:\(a_i\), \(b_i\) 和 \(c_i\).定义一个元素的 ...