题目链接:

http://poj.org/problem?id=1185

炮兵阵地

Time Limit: 2000MS
Memory Limit: 65536K
#### 问题描述
> 司令部的将军们打算在`N*M`的网格地图上部署他们的炮兵部队。一个`N*M`的地图由N行M列组成,地图的每一格可能是山地(用"H" 表示),也可能是平原(用"P"表示),如下图。在每一格平原地形上最多可以布置一支炮兵部队(山地上不能够部署炮兵部队);一支炮兵部队在地图上的攻击范围如图中黑色区域所示:
> 如果在地图中的灰色所标识的平原上部署一支炮兵部队,则图中的黑色的网格表示它能够攻击到的区域:沿横向左右各两格,沿纵向上下各两格。图上其它白色网格均攻击不到。从图上可见炮兵的攻击范围不受地形的影响。
> 现在,将军们规划如何部署炮兵部队,在防止误伤的前提下(保证任何两支炮兵部队之间不能互相攻击,即任何一支炮兵部队都不在其他支炮兵部队的攻击范围内),在整个地图区域内最多能够摆放多少我军的炮兵部队。
>
#### 输入
> 第一行包含两个由空格分割开的正整数,分别表示N和M;
> 接下来的N行,每一行含有连续的M个字符('P'或者'H'),中间没有空格。按顺序表示地图中每一行的数据。N 输出

仅一行,包含一个整数K,表示最多能摆放的炮兵部队的数量。

####样例输入

5 4
PHPP
PPHH
PPPP
PHPP
PHHP

####样例输出

6

题解

第一感觉就是转移和前两行的状态有关,这样就变成2^20了,转移的时候会直接炸掉。

正解是,只记录有效的状态,这样发现一行有效的状态就60个,两行的也就3600,然后就可以直接按行转移了。

代码

#include<map>
#include<set>
#include<cmath>
#include<queue>
#include<stack>
#include<ctime>
#include<vector>
#include<cstdio>
#include<string>
#include<bitset>
#include<cstdlib>
#include<cstring>
#include<iostream>
#include<algorithm>
#include<functional>
using namespace std;
#define X first
#define Y second
#define mkp make_pair
#define lson (o<<1)
#define rson ((o<<1)|1)
#define mid (l+(r-l)/2)
#define sz() size()
#define pb(v) push_back(v)
#define all(o) (o).begin(),(o).end()
#define clr(a,v) memset(a,v,sizeof(a))
#define bug(a) cout<<#a<<" = "<<a<<endl
#define rep(i,a,b) for(int i=a;i<(b);i++)
#define scf scanf
#define prf printf typedef long long LL;
typedef vector<int> VI;
typedef pair<int,int> PII;
typedef vector<pair<int,int> > VPII; const int INF=0x3f3f3f3f;
const LL INFL=10000000000000000LL;
const double eps=1e-9; const double PI = acos(-1.0); //start---------------------------------------------------------------------- const int maxn=111;
const int maxs=66;
int n,m; ///存有效状态
VI mysta;
vector<LL> sumv; ///dp[cur][i][j]现在考虑前cur行,前1行状态是i,第cur行状态是j,能塞下的最多炮兵。
LL dp[maxn][maxs][maxs]; ///预处理出所有未冲突的状态
void pre() {
for(int i=0; i<(1<<10); i++) {
int cnt=0;
int mi=INF,pre=-100;
for(int j=0; j<10; j++) {
if(i&(1<<j)) {
cnt++;
if(mi>j-pre) mi=j-pre;
pre=j;
}
}
if(mi<=2) continue;
mysta.pb(i);
sumv.pb(cnt);
}
} char str[maxn][22];
int tot; ///判断有没有吧炮搭到山上
bool ok(int x,int i) {
for(int j=0; j<m; j++) {
if(!(x&(1<<j))) continue;
if(str[i][j]=='H') {
return false;
}
}
return true;
} ///判断上方冲突
bool ok2(int pre,int cur) {
for(int i=0; i<m; i++) {
if((cur&(1<<i))==0) continue;
if(pre&(1<<i)) return false;
}
return true;
} ///特判只有一行的情况
void solve1() {
LL ans=0;
for(int i=0; i<tot; i++) {
if(ok(mysta[i],0)) ans=max(ans,sumv[i]);
}
prf("%lld\n",ans);
} int main() {
pre();
while(scf("%d%d",&n,&m)==2&&n) {
tot=upper_bound(all(mysta),(1<<m)-1)-mysta.begin();
rep(i,0,n) scf("%s",str[i]);
if(n==1) {
solve1();
continue;
} ///预处理出前两行
clr(dp,0);
for(int i=0; i<tot; i++) {
if(!ok(mysta[i],0)) continue;
for(int j=0; j<tot; j++) {
if(!ok(mysta[j],1)) continue;
if(!ok2(mysta[i],mysta[j])) continue;
dp[1][i][j]=sumv[i]+sumv[j];
}
} ///递推
for(int cur=2; cur<n; cur++) {
for(int k=0; k<tot; k++) {
if(!ok(mysta[k],cur)) continue;
for(int i=0; i<tot; i++) {
if(!ok2(mysta[i],mysta[k])) continue;
for(int j=0; j<tot; j++) {
if(!ok2(mysta[j],mysta[k])) continue;
dp[cur][j][k]=max(dp[cur][j][k],dp[cur-1][i][j]+sumv[k]);
}
}
}
} LL ans=0;
for(int i=0; i<tot; i++) {
for(int j=0; j<tot; j++) {
ans=max(ans,dp[n-1][i][j]);
}
} prf("%lld\n",ans);
}
return 0;
} //end-----------------------------------------------------------------------

Notes

当上帝关上一扇门的同时,会给你开启新的一扇门。
这题相比方格取数虽然状态变成两行了,但是有效的状态却变少了。

POJ 1185 炮兵阵地 状压dp的更多相关文章

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

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

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

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

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

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

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

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

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

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

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

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

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

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

  8. luogu 2704 炮兵阵地 状压dp

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

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

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

随机推荐

  1. EasyX_无法填充圆颜色的问题

    官网:https://www.easyx.cn/ 在线帮助文档:https://docs.easyx.cn/ 目标:生成一个边框为黄色,填充为蓝色的圆 遇到的问题:使用以下代码,只能生成边框为黄色的圆 ...

  2. rtthread移植到jz2440之BootLoader

    从2016年第一次接触rtthread,感觉很容易上手,记得一个项目是小飞行器上的IPC,趁着空闲,手里有一块jz2440的板子,准备在这块板子上跑起来rtthread,查了很多资料,最后决定自己写一 ...

  3. WPF的IsSynchronizedWithCurrentItem属性

    如果两个控件都绑定到同一个源(ObservableCollection)集合视图时,该对象会自动绑定到该视图的 CurrentItem.请注意,CollectionViewSource 对象会自动同步 ...

  4. inux下进程的最大线程数、进程最大数、进程打开的文件数

    inux下进程的最大线程数.进程最大数.进程打开的文件数 2008-12-07 23:48 =========================    如下转载自这里. linux 系统中单个进程的最大 ...

  5. 2-4 list练习题

    参考答案 >>> names = [] >>> names.append('old_driver') >>> names.append('rain ...

  6. 基于Vue的弹框实例

    看到博客的人,请养成写博客的习惯,不会不会,就怕曾经会过,现在想不起来了,一起加油....................  让学习真的成为一种习惯,同时要注意身体 <!DOCTYPE html ...

  7. @Helper辅助方法和@functions自定义函数

    1.首先说下@helper辅助方法,当我们在多个视图中共用相同的方法的时候,可以把此方法剥离出来放到一个位置,此时就可以用到@Helper辅助方法,首先我们在解决方案右键添加 App_Code文件夹, ...

  8. 负载均衡@StackExchange.Redis实现Session外置--纯干货喂饱你

    Redis和StackExchange.Redis redis有多个数据库1.redis 中的每一个数据库,都由一个 redisDb 的结构存储.其中,redisDb.id 存储着 redis 数据库 ...

  9. 【Python学习笔记之三】lambda表达式用法小结

    除了def语句之外,Python还提供了一种生成函数对象的表达式形式.由于它与LISP语言中的一个工具很相似,所以称为lambda.就像def一样,这个表达式创建了一个之后能够调用的函数,但是它返回了 ...

  10. 【转载】钉钉开发c#帮助类 获取用户信息 DingHelper.cs

    using System;using System.Collections.Generic;using System.Configuration;using System.Linq;using Sys ...