POJ:1185-炮兵阵地(状压dp入门)
炮兵阵地
Time Limit: 2000MS Memory Limit: 65536K
Description
司令部的将军们打算在N*M的网格地图上部署他们的炮兵部队。一个N*M的地图由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
解题心得:
- 状压dp,很神奇啊,一般状压dp的标志就是某一数据量比较小,例如,行的长度很大,列的长度在15以下,可以直接使用二进制01表示状态,然后化为十进制,用一个数来表示每一行的状态。
- 这个题其实也可以说是二进制枚举,用一个二进制数来表示一行的状态,筛出一些不符合要求的数,例如,同一行中两个炮台在一行之中距离小于3,这样就可以筛去一大部分数(1000个数可以筛去930个左右)。然后枚举每一行的可以安排炮塔的状态,枚举的时候检验是否和前面已经安排的炮塔产生冲突,如果不产生冲突,那么记录下不产生冲突的情况下可以安排的最多的炮塔的数量。
- 状态转移是在从前面安排好的炮塔的状态来转移出下面可以安排的炮塔,并且既要符合前面安排好的炮塔,还要得到可以安排炮塔的最大值。dp[i][j][k]代表的是前面i行最多可以安排k个炮塔,前i-1行可以安排j个炮塔。转移方程式为:
- dp[i][j2][j1] = max(dp[i][j2][j1],dp[i-1][j3][j2]+sum[j1])
- 有点复杂,不懂得可以看看实现的方法。
#include<stdio.h>
#include<cstring>
#include<algorithm>
using namespace std;
const int maxn = 110;
int maps[maxn],sum[maxn],stk[maxn],cnt,dp[maxn][maxn][maxn];
char s[maxn][maxn];
int n,m;
void init()
{
cnt = 0;
memset(dp,-1,sizeof(dp));
memset(maps,0,sizeof(maps));
memset(sum,0,sizeof(sum));
memset(stk,0,sizeof(stk));
for(int i=0;i<n;i++)
{
scanf("%s",s[i]);
for(int j=0;j<m;j++)
if(s[i][j] == 'H')
maps[i] |= (1<<(m-j-1));//用一个数来表示地图上山峰的情况
}
}
bool ok(int x)//这个数是否符合一行中安排炮塔的条件(每个炮塔距离大于2)
{
if(x&(x<<1))
return false;
if(x&(x<<2))
return false;
return true;
}
int get_sum(int x)//记录这个数代表的状态中有多少个炮塔
{
int sum = 0;
while(x)
{
if(x & 1)
sum++;
x>>=1;
}
return sum;
}
void get_stk()
{
for(int i=0;i<(1<<m);i++)
if(ok(i))//检验这个数是否符合一行中安排炮塔的条件(每个炮塔的距离大于2)
{
stk[cnt] = i;
sum[cnt++] = get_sum(i);//记录这个状态中有多少个炮塔
}
}
int get_ans()
{
for(int i=0;i<cnt;i++)
if(!(maps[0] & stk[i]))//第一行比价特殊只要不和地图冲突就可以了
dp[0][0][i] = sum[i];
for(int i=1;i<n;i++)//安排每一行
for(int j1=0;j1<cnt;j1++)//枚举每一行安排炮塔的情况
{
if(maps[i]&stk[j1])//枚举的当前状态是否和地图上的山峰冲突
continue;
for(int j2=0;j2<cnt;j2++)//枚举的当前状态是否和上一行冲突
{
if(stk[j2] & stk[j1])
continue;
for(int j3=0;j3<cnt;j3++)//枚举的当前状态是否和上上行冲突
{
if(stk[j3] & stk[j1])
continue;
dp[i][j2][j1] = max(dp[i][j2][j1],dp[i-1][j3][j2]+sum[j1]);//找出不冲突的最大炮塔数量
}
}
}
int Max = 0;
for(int i=0;i<cnt;i++)
for(int j=0;j<cnt;j++)
if(dp[n-1][i][j] > Max)
Max = dp[n-1][i][j];
return Max;
}
int main()
{
while(~scanf("%d%d",&n,&m))
{
init();//初始化各值
get_stk();
int ans = get_ans();
printf("%d\n",ans);
}
return 0;
}
POJ:1185-炮兵阵地(状压dp入门)的更多相关文章
- POJ 1185 炮兵阵地 状压dp
题目链接: http://poj.org/problem?id=1185 炮兵阵地 Time Limit: 2000MS Memory Limit: 65536K 问题描述 司令部的将军们打算在N*M ...
- POJ 1185炮兵阵地 (状压DP)
题目链接 POJ 1185 今天艾教留了一大堆线段树,表示做不动了,就补补前面的题.QAQ 这个题,我第一次写还是像前面HDU 2167那样写,发现这次影响第 i 行的还用i-2行那样,那以前的方法就 ...
- poj - 1185 炮兵阵地 状压DP 解题报告
炮兵阵地 Time Limit: 2000MS Memory Limit: 65536K Total Submissions: 21553 Accepted: 8363 Description ...
- [poj 1185] 炮兵阵地 状压dp 位运算
Description 司令部的将军们打算在N*M的网格地图上部署他们的炮兵部队.一个N*M的地图由N行M列组成,地图的每一格可能是山地(用"H" 表示),也可能是平原(用&quo ...
- POJ1185 炮兵阵地 —— 状压DP
题目链接:http://poj.org/problem?id=1185 炮兵阵地 Time Limit: 2000MS Memory Limit: 65536K Total Submissions ...
- TZOJ 4912 炮兵阵地(状压dp)
描述 司令部的将军们打算在N*M的网格地图上部署他们的炮兵部队.一个N*M的地图由N行M列组成,地图的每一格可能是山地(用"H" 表示),也可能是平原(用"P" ...
- 洛谷P2704 [NOI2001]炮兵阵地 [状压DP]
题目传送门 炮兵阵地 题目描述 司令部的将军们打算在N*M的网格地图上部署他们的炮兵部队.一个N*M的地图由N行M列组成,地图的每一格可能是山地(用“H” 表示),也可能是平原(用“P”表示),如下图 ...
- [POJ1185][NOI2001]炮兵阵地 状压DP
题目链接:http://poj.org/problem?id=1185 很裸的状压,考虑对于一行用二进制储存每一种的状态,但是状态太多了做不了. 观察到有很多状态都是不合法的,于是我们预处理出合法的状 ...
- luogu 2704 炮兵阵地 状压dp
状压的基础题吧 第一次看感觉难上天,后来嘛就.. 套路:先根据自身状态筛出可行状态,再根据地图等其他限制条件筛选适合的状态加入答案 f i,j,k 分别代表 行数,本行状态,上行状态,再累加答案即可 ...
- [NOI2001]炮兵阵地 状压DP
题面: 司令部的将军们打算在N*M的网格地图上部署他们的炮兵部队.一个N*M的地图由N行M列组成,地图的每一格可能是山地(用“H” 表示),也可能是平原(用“P”表示),如下图.在每一格平原地形上最多 ...
随机推荐
- (转)Linux基础知识学习
Linux基础知识学习 原文:http://blog.csdn.net/ye_wei_yang/article/details/52777499 一.Linux的磁盘分区及目录 Linux的配置是通过 ...
- windows无法启动redis服务,错误码1067
https://blog.csdn.net/kissdead0xzy/article/details/84332870
- JS filter使用
filter 用于筛选数组中符合条件的所以元素,filter只能接受函数 注意:filter只返回筛选结果,不会对原来数组改变 实现方法: <html lang="en"&g ...
- MVC FileResult
你如何将文件传送给用户取决于你最开始如何存储它,如果你将文件存入数据库,你会用流的方式将文件返还给用户,如果你将文件存在硬盘中,你只需要提供一个超链接即可,或者也可以以流的方式.每当你需要以流的方式将 ...
- .net的基础知识点
在这个It市场都是风云变化的,都是又市场供需来定的,当年iOS火的一塌糊涂的,现在也出现找不到工作的,满地的出入门者,我就属于其中一个,在一个逼不得已的情况下,开始转行做.net ,我相信当年的很多. ...
- 发现了一个entity framework的BUG
小弟学浅才疏可能是小题大做,但遇上了并且让我麻烦了一阵,就值得记下来 BUG的过程就是我在建立实体模型的时候 命名了一个叫system的实体模型 导致了所有生成类中 引用using system失败
- Git操作(及操作github)
一.git操作: 1.创建一个项目: 2.打开项目,在项目目录下右键单击--打开 git bash: 3.初始化项目仓库 创建项目仓库 命令: git init ; git status 查看文 ...
- 定时器 & 日期时间对象 & 正则
1 JavaScript 计时事件 通过使用 JavaScript,有能力做到在一个设定的时间间隔之后来执行代码,而不是在函数被调用后立即执行,这称之为计时事件. 两个关键方法是: setInterv ...
- pixhawk 固件Firmware内执行make px4fmu-v2_default 编译报错解决办法
执行下列指令报错 make px4fmu-v2_default /bin/sh: 1: Tools/check_cmake.sh: Permission denied Makefile:44: Not ...
- 零基础逆向工程23_PE结构07_重定位表_IAT表(待补充)
重定位表 待补充 IAT表 待补充