炮兵阵地

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


解题心得:

  1. 状压dp,很神奇啊,一般状压dp的标志就是某一数据量比较小,例如,行的长度很大,列的长度在15以下,可以直接使用二进制01表示状态,然后化为十进制,用一个数来表示每一行的状态。
  2. 这个题其实也可以说是二进制枚举,用一个二进制数来表示一行的状态,筛出一些不符合要求的数,例如,同一行中两个炮台在一行之中距离小于3,这样就可以筛去一大部分数(1000个数可以筛去930个左右)。然后枚举每一行的可以安排炮塔的状态,枚举的时候检验是否和前面已经安排的炮塔产生冲突,如果不产生冲突,那么记录下不产生冲突的情况下可以安排的最多的炮塔的数量。
  3. 状态转移是在从前面安排好的炮塔的状态来转移出下面可以安排的炮塔,并且既要符合前面安排好的炮塔,还要得到可以安排炮塔的最大值。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])
  4. 有点复杂,不懂得可以看看实现的方法。

#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入门)的更多相关文章

  1. POJ 1185 炮兵阵地 状压dp

    题目链接: http://poj.org/problem?id=1185 炮兵阵地 Time Limit: 2000MS Memory Limit: 65536K 问题描述 司令部的将军们打算在N*M ...

  2. POJ 1185炮兵阵地 (状压DP)

    题目链接 POJ 1185 今天艾教留了一大堆线段树,表示做不动了,就补补前面的题.QAQ 这个题,我第一次写还是像前面HDU 2167那样写,发现这次影响第 i 行的还用i-2行那样,那以前的方法就 ...

  3. poj - 1185 炮兵阵地 状压DP 解题报告

    炮兵阵地 Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 21553   Accepted: 8363 Description ...

  4. [poj 1185] 炮兵阵地 状压dp 位运算

    Description 司令部的将军们打算在N*M的网格地图上部署他们的炮兵部队.一个N*M的地图由N行M列组成,地图的每一格可能是山地(用"H" 表示),也可能是平原(用&quo ...

  5. POJ1185 炮兵阵地 —— 状压DP

    题目链接:http://poj.org/problem?id=1185 炮兵阵地 Time Limit: 2000MS   Memory Limit: 65536K Total Submissions ...

  6. TZOJ 4912 炮兵阵地(状压dp)

    描述 司令部的将军们打算在N*M的网格地图上部署他们的炮兵部队.一个N*M的地图由N行M列组成,地图的每一格可能是山地(用"H" 表示),也可能是平原(用"P" ...

  7. 洛谷P2704 [NOI2001]炮兵阵地 [状压DP]

    题目传送门 炮兵阵地 题目描述 司令部的将军们打算在N*M的网格地图上部署他们的炮兵部队.一个N*M的地图由N行M列组成,地图的每一格可能是山地(用“H” 表示),也可能是平原(用“P”表示),如下图 ...

  8. [POJ1185][NOI2001]炮兵阵地 状压DP

    题目链接:http://poj.org/problem?id=1185 很裸的状压,考虑对于一行用二进制储存每一种的状态,但是状态太多了做不了. 观察到有很多状态都是不合法的,于是我们预处理出合法的状 ...

  9. luogu 2704 炮兵阵地 状压dp

    状压的基础题吧 第一次看感觉难上天,后来嘛就.. 套路:先根据自身状态筛出可行状态,再根据地图等其他限制条件筛选适合的状态加入答案 f i,j,k 分别代表 行数,本行状态,上行状态,再累加答案即可 ...

  10. [NOI2001]炮兵阵地 状压DP

    题面: 司令部的将军们打算在N*M的网格地图上部署他们的炮兵部队.一个N*M的地图由N行M列组成,地图的每一格可能是山地(用“H” 表示),也可能是平原(用“P”表示),如下图.在每一格平原地形上最多 ...

随机推荐

  1. STM32开关总中断

    引用 http://www.amobbs.com/forum.php?mod=viewthread&tid=5397451 在 STM32/Cortex-M3 中是通过改变 CPU 的当前优先 ...

  2. LinkedList源码及原理

    简介 内部结构分析 LinkedList源码分析 构造方法 添加(add)方法 根据位置取数据的方法 根据对象得到索引的方法 检查链表是否包含某对象的方法: 删除(remove/pop)方法 Link ...

  3. 给Visual Studio更换皮肤和背景图

    1.先安装更换皮肤的插件 VS菜单栏里面找到:工具>扩展和更新>联机>搜索: Theme Editor 下载并安装: 安装后先不着急重启VS 然后,安装可更改背景图片的插件:工具&g ...

  4. 滚动条插件mCustomScrollbar

    1. 参考网站 http://www.wufangbo.com/mcustomscrollbar/ 2. 使用方法

  5. UIcollectionView 实现 轮番图

    UICollectionView 用作轮番图的实现,demo 地址:https://github.com/SummerHH/YJCYCleCollectionVIew #import <UIKi ...

  6. 准备Kendo UI 开发环境

    准备 首先你需要从 Telerik 网站下载试用版开发包,注意需要注册后才能下载. 下载后直接解压后包含下面几个文件和目录: ./examples – 示例. /js – minified 化后的 J ...

  7. python-day1作业(感谢视频老师留的作业)

    __author__ = 'zht' #!/usr/bin/env python # -*- coding: utf-8 -*- ''' #努力学习每一天 ''' #尝试次数计数器 tries = 0 ...

  8. Redis、Memcache区别

    Redis.Memcache区别 redis单核 memcahce多核 redis支持数据持久化 redis支持的数据类型比较多 memcache 只有key->value类型 key-> ...

  9. SQL查询-约束-多表

    一.SQL语句查询     1.聚合函数         COUNT()函数,统计表中记录的总数量         注:COUNT()返回的值为Long类型;可通过Long类的intValue()方法 ...

  10. cpu 满载测试软件

    for i in `seq 1 $(cat /proc/cpuinfo |grep "physical id" |wc -l)`; do dd if=/dev/zero of=/d ...