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,把每一行看做一个状态,要除去同一 ...
随机推荐
- Window下JDK安装教程
1.准备 win10系统,其他windows系统安装过程大同小异官网下载jdk1.8下载地址:https://www.oracle.com/technetwork/java/javase/downlo ...
- 微信小程序组件 模块化错和叹号
wxml 页面 <import src="/pages/lianxi/lianxi.wxml" /> //引入文件 <view style='position: ...
- three.js:Failed to execute 'texImage2D' on 'WebGLRenderingContext解决方案
three.js加载图片时,出现Failed to execute 'texImage2D' on 'WebGLRenderingContext .Tainted canvases may not b ...
- HTML之绝对路径与相对路径
路径指文件存放的位置,在网页中利用路径可以引用文件,插入图像.视频等.表示路径的方法有两种:相对路径,绝对路径.以下讨论均是在HTML环境下进行. 相对路径 相对路径是指目标相对于当前文件的路径,网页 ...
- Java的StringBuIlder扩容机制
JDK 1.6中,扩容的源码是这样: void expandCapacity(int minimumCapacity) { int newCapacity = (value.length + 1) * ...
- lepus天兔数据库监控
本篇文章的前提是服务器装了mysql服务.git,我这边就不写出来了,自行百度,装下mysql服务,比较简单 一.安装LAMP基础环境 Xampp下载地址:https://www.apachefrie ...
- nginx配置虚拟路径下载文件(.apk)
公司将安卓apk文件放到服务器上,实现用户点击链接并下载 nginx version 1.14.1 nginx配置修改 server { listen 80; server_name localhos ...
- 【原创】c# Winform 使用 web 的UrlEncode/UrlDecode 方法
1.先 右键引用,添加 System.Web 数据集 2.语句如下 string s= System.Web.HttpUtility.UrlEncode("123", System ...
- BZOJ 2844: albus就是要第一个出场
2844: albus就是要第一个出场 Time Limit: 6 Sec Memory Limit: 128 MBSubmit: 1134 Solved: 481[Submit][Status] ...
- 【BZOJ1041】圆上的整点(数论)
[BZOJ1041]圆上的整点(数论) 题面 BZOJ 洛谷 题解 好神仙的题目啊. 安利一个视频,大概是第\(7\)到\(19\)分钟的样子 因为要质因数分解,所以复习了一下\(Pollard\_r ...