【简●解】POJ 1185,LG P2704【炮兵阵地】
【传送门】
【题目大意】
司令部的将军们打算在 \(N\times M\) 的网格地图上部署他们的炮兵部队。一个 \(N\times M\) 的地图由 \(N\) 行 \(M\) 列组成,地图的每一格可能是山地(用 H
表示),也可能是平原(用 P
表示),如下图。在每一格平原地形上最多可以布置一支炮兵部队(山地上不能够部署炮兵部队);一支炮兵部队在地图上的攻击范围如图中黑色区域所示:
求最多能摆放的炮兵部队的数量。
【关键词】
- 状压DP
- 预处理
- 进制运算
【分析】
可以说这道题是状压\(DP\)的经典例题了,我们思考,每一行的状态只会被当前这一行的地形和前两行的状态束缚,所以我们可以预处理第一行与第二行,然后枚举每一行的状态进行\(DP\)转移。
细节方面就看代码吧。
【Code】
//#include<bits/stdc++.h>
#include<cmath>
#include<ctime>
#include<queue>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<iostream>
#include<algorithm>
#define debug() puts("FBI WARNING!")
#define ll long long
using namespace std;
inline int read(){
int f = 1, x = 0;char ch;
do { ch = getchar(); if (ch == '-') f = -1; } while (ch < '0'||ch>'9');
do {x = x*10+ch-'0'; ch = getchar(); } while (ch >= '0' && ch <= '9');
return f*x;
}
int n, m, M;
int maps[105], num[(1<<11)], c[80], len;
char tmp[15];
int dp[105][80][80], ans;
inline bool judge(int a, int b) {
if ((a & b) == 0) return 1;
else return 0;
}
inline bool can(int x) {
if (((x << 1) & x) || (x << 2) & x || (x >> 1) & x || (x >> 2) & x) return 0;
else return 1;
}
inline void init() {
/*
预处理num[]
*/
for (int i = 1;i < M; ++i) {
int x = i;
while (x) {
num[i]+=x%2;
x >>= 1;
}
}
/*
预处理c[]
*/
for (int i = 0;i < M; ++i) {
if (can(i)) {
c[++len] = i;
}
}
}
int main(){
n = read(); m = read();
M = (1 << m);
init();
for (int i = 1;i <= n; ++i) {
scanf("%s", tmp);
for (int j = 0;j < m; ++j)
if (tmp[j] == 'H')
maps[i] += (1 << j);
}
/*for (int i = 0;i < n; ++i) {
printf("%d\n", maps[i]);
}*/
for (int i = 1;i <= len; ++i) { // 第一行处理
if (judge(c[i], maps[1]))
dp[1][i][1] = num[c[i]];
}
/*for (int i = 0;i < M; ++i) {
printf("dp[0][%d][0] = %d can:%d, judge:%d\n", i, dp[0][i][0], can(i), judge(i, maps[0]));
}*/
for (int i = 1;i <= len; ++i) { // 第二行处理
if (judge(c[i], maps[2]))
for (int j = 1;j <= len; ++j) {
if (judge(c[j], maps[1]) && judge(c[i], c[j]))
dp[2][i][j] = max(dp[2][i][j], dp[1][j][1]+num[c[i]]);
}
}
for (int i = 3;i <= n; ++i) { // DP
for (int j = 1;j <= len; ++j) {
if (judge(c[j], maps[i]))
for (int k = 1;k <= len; ++k) {
if (judge(c[j], c[k]) && judge(c[k], maps[i-1])) {
for (int s = 1;s <= len; ++s) {
if (judge(c[s], c[j]) && judge(c[s], c[k]) && judge(c[s], maps[i-2])) {
dp[i][j][k] = max(dp[i][j][k], dp[i-1][k][s]+num[c[j]]);
}
}
}
}
}
}
for (int i = 1;i <= len; ++i) { //求解答案
for (int j = 1;j <= len; ++j) {
ans = max(ans, dp[n][i][j]);
}
}
printf("%d", ans);
return 0;
}
【简●解】POJ 1185,LG P2704【炮兵阵地】的更多相关文章
- P2704 炮兵阵地
题目描述 司令部的将军们打算在N*M的网格地图上部署他们的炮兵部队.一个N*M的地图由N行M列组成,地图的每一格可能是山地(用“H” 表示),也可能是平原(用“P”表示),如下图.在每一格平原地形上最 ...
- [LUOGU] P2704 炮兵阵地
题目描述 司令部的将军们打算在N*M的网格地图上部署他们的炮兵部队. 一个N*M的地图由N行M列组成,地图的每一格可能是山地(用"H" 表示), 也可能是平原(用"P&q ...
- luogu P2704 炮兵阵地(经典状态压缩DP)
方格有m*n个格子,一共有2^(m+n)种排列,很显然不能使用暴力法,因而选用动态规划求解. 求解DP问题一般有3步,即定义出一个状态 求出状态转移方程 再用算法实现.多数DP题难youguan点在于 ...
- 洛谷P2704 炮兵阵地
本题过于经典...... 对于这种网格状压DP,套路一波刷表法DFS转移就没了. 三进制状压,0表示当前,上一个都没有.1表示当前无,上一个有.2表示当前有. 转移的条件就是上一行为0,当前不是山地, ...
- 【Luogu】P2704炮兵阵地(状压DP)
题目链接 话说还真没见过能影响两行的状压.想了半天想出来f数组再多一维就能表示,但是没想到怎么才能不爆空间…… 也是从这道题里学到的一个妙招. 可以把合法状态存到一个数组里,然后用数组下标来映射状态. ...
- POJ 1185 炮兵阵地 状压dp
题目链接: http://poj.org/problem?id=1185 炮兵阵地 Time Limit: 2000MS Memory Limit: 65536K 问题描述 司令部的将军们打算在N*M ...
- POJ 1185 炮兵阵地(状压DP)
炮兵阵地 Time Limit: 2000MS Memory Limit: 65536K Total Submissions: 26426 Accepted: 10185 Descriptio ...
- POJ 1185 状态压缩DP 炮兵阵地
题目直达车: POJ 1185 炮兵阵地 分析: 列( <=10 )的数据比较小, 一般会想到状压DP. Ⅰ.如果一行10全个‘P’,满足题意的状态不超过60种(可手动枚举). Ⅱ.用DFS ...
- POJ 1185 炮兵阵地(动态规划+状态压缩)
炮兵阵地 Description 司令部的将军们打算在N*M的网格地图上部署他们的炮兵部队.一个N*M的地图由N行M列组成,地图的每一格可能是山地(用"H" 表示),也可能是平原( ...
随机推荐
- E20180426-hm
transition n. 过渡,转变,变迁; [语] 转换; [乐] 变调 flip vt. 按(开关); 快速翻转; 急挥; n. 空翻; 浏览; (射击时枪管的) 跳跃; 轻抛; win ...
- Cannot call value of non-function type 'UITextView'报错
iOS里面的开发,类写到一半就报错这个.后来发现是因为重名的时候召唤对象不明确的问题.先贴代码,晚点再说 出错点 //ヒントをクリアするためのイニシャライザ init (clearStr: UITex ...
- 【WIP】swift3的timer的用法
创建: 2017/10/14 更新: 2017/10/14 标题加上[WIP],补充创建时间 回家再写
- iOS开发检测是否开启定位、是否允许消息推送等权限
1.iOS开发检测是否开启定位: 需要导入: #import <CoreLocation/CoreLocation.h> 代码如下: + (void)openLocationService ...
- IT兄弟连 JavaWeb教程 Servlet
Servlet的定义 Java Servlet是运行在Web服务器或应用服务器上的程序,它是作为来自Web浏览器或其他HTTP客户端的请求和HTTP服务器上的数据库或应用程序之间的中间层. 使用Ser ...
- 分布式集群环境下,如何实现session共享三(环境搭建)
这是分布式集群环境下,如何实现session共享系列的第三篇.在上一篇:分布式集群环境下,如何实现session共享二(项目开发)中,准备好了一个通过原生态的servlet操作session的案例.本 ...
- SpringBoot的配置文件加载顺序以及如何获取jar包里的资源路径
一.读取配置文件的四种方式 这四种配置文件放置方式的读取优先级依次递减,具体可以查看官方文档. 1.1jar包同级目录下的config文件夹里的配置文件 其实我以前就见过这种方式了,只是不知道怎么做的 ...
- 怎么让普通用户达到root用户也可以拥有权限修改文件(CentOS系统)
为什么会提出这个问题呢? 答:因为啊,比如我们在大数据集群搭建的时候,经常会需要配置环境变量,如/etc/profile.那么,问题来了,每次必须得都切换到root用户才能,但是呢,如何可以在普通用户 ...
- tarjan有向图的强连通
强连通:在有向图G中,两个顶点间至少存在一条路径,则两个点强连通. 强连通图:在有向图中,每两个顶点都强连通,则有向图G就是一个强连通图. 强连通分量:在非强连通图中的极大强连通子图,就称为强连通分量 ...
- Python之单元测试——HTMLTestRunner
前置条件:把HTMLTestRunner.py文件拷贝到External Libraries—>site-packages里面 import unittestimport HTMLTestRun ...