POJ1185 炮兵阵地 —— 状压DP
题目链接:http://poj.org/problem?id=1185
Time Limit: 2000MS | Memory Limit: 65536K | |
Total Submissions: 30608 | Accepted: 11828 |
Description

如果在地图中的灰色所标识的平原上部署一支炮兵部队,则图中的黑色的网格表示它能够攻击到的区域:沿横向左右各两格,沿纵向上下各两格。图上其它白色网格均攻击不到。从图上可见炮兵的攻击范围不受地形的影响。
现在,将军们规划如何部署炮兵部队,在防止误伤的前提下(保证任何两支炮兵部队之间不能互相攻击,即任何一支炮兵部队都不在其他支炮兵部队的攻击范围内),在整个地图区域内最多能够摆放多少我军的炮兵部队。
Input
接下来的N行,每一行含有连续的M个字符('P'或者'H'),中间没有空格。按顺序表示地图中每一行的数据。N <= 100;M <= 10。
Output
Sample Input
- 5 4
- PHPP
- PPHH
- PPPP
- PHPP
- PHHP
Sample Output
- 6
Source
题解:
1.学了轮廓线更新之后,一看到这种棋盘DP就忍不住往轮廓线上想:由于状态转移关系到上两行,所以轮廓线就需要记录两行了,即每一列都需要记录两个格,可知每一列有三种状态,那么用四进制表示。所以复杂度大概为:100*10*(4^10)*2= 很大很大。轮廓线更新都满足不了复杂度要求,那行更新就更加不能满足了吧?能,看下一点。
2.枚举一行的所有状态,因为要求两点间距离>2,所以可以筛掉很多无效状态,最终得到60个有效状态。
3.得到有效状态后,剩下的工作就是典型的状压DP行更新了。复杂度:100*64*64*64。
代码如下:
- #include <iostream>
- #include <cstdio>
- #include <cstring>
- #include <algorithm>
- #include <vector>
- #include <cmath>
- #include <queue>
- #include <stack>
- #include <map>
- #include <string>
- #include <set>
- using namespace std;
- typedef long long LL;
- const double EPS = 1e-;
- const int INF = 2e9;
- const LL LNF = 9e18;
- const int MOD = 1e5;
- const int MAXN = 1e3+;
- //row记录每一行的可放状态
- //sta记录有效状态,cnt记录有效状态有多少个点
- //dp[i][j][k]:更新到第i行,且第i-1行的状态为sta[j],第i行的状态为sta[k] 的最大放置数。
- int row[], sta[], cnt[MAXN], dp[][][];
- int main()
- {
- int n, m = ;
- char str[];
- while(scanf("%d%d", &n,&m)!=EOF)
- {
- for(int i = ; i<=n; i++)
- {
- scanf("%s", str);
- row[i] = ;
- for(int j = ; j<m; j++)
- row[i] += (str[j]=='H')*(<<j);
- }
- int tot = ;
- for(int s = ; s<(<<m); s++)
- {
- if( !(s&(s<<)) && !(s&(s<<)) )
- {
- sta[++tot] = s;
- cnt[tot] = ;
- for(int j = ; j<m; j++)
- if(s&(<<j))
- cnt[tot]++;
- }
- }
- memset(dp, , sizeof(dp));
- for(int i = ; i<=n; i++)
- {
- for(int j = ; j<=tot; j++)
- {
- int s1 = sta[j];
- if(i> && (s1&row[i-])) continue;
- for(int k = ; k<=tot; k++)
- {
- int s2 = sta[k];
- if(i> && (s2&row[i-])) continue;
- if(s1&s2) continue;
- for(int t = ; t<=tot; t++)
- {
- int s3 = sta[t];
- if(s3&row[i]) continue;
- if((s1&s3)||(s2&s3)) continue;
- dp[i][k][t] = max(dp[i][k][t], dp[i-][j][k]+cnt[t]);
- }
- }
- }
- }
- int ans = ;
- for(int i = ; i<=tot; i++)
- for(int j = ; j<=tot; j++)
- ans = max(ans, dp[n][i][j]);
- printf("%d\n", ans);
- }
- }
POJ1185 炮兵阵地 —— 状压DP的更多相关文章
- poj1185 炮兵阵地 状压dp
司令部的将军们打算在N*M的网格地图上部署他们的炮兵部队.一个N*M的地图由N行M列组成,地图的每一格可能是山地(用"H" 表示),也可能是平原(用"P"表示) ...
- 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 很裸的状压,考虑对于一行用二进制储存每一种的状态,但是状态太多了做不了. 观察到有很多状态都是不合法的,于是我们预处理出合法的状 ...
- poj - 1185 炮兵阵地 状压DP 解题报告
炮兵阵地 Time Limit: 2000MS Memory Limit: 65536K Total Submissions: 21553 Accepted: 8363 Description ...
- luogu 2704 炮兵阵地 状压dp
状压的基础题吧 第一次看感觉难上天,后来嘛就.. 套路:先根据自身状态筛出可行状态,再根据地图等其他限制条件筛选适合的状态加入答案 f i,j,k 分别代表 行数,本行状态,上行状态,再累加答案即可 ...
- POJ 1185 炮兵阵地 状压dp
题目链接: http://poj.org/problem?id=1185 炮兵阵地 Time Limit: 2000MS Memory Limit: 65536K 问题描述 司令部的将军们打算在N*M ...
- [NOI2001]炮兵阵地 状压DP
题面: 司令部的将军们打算在N*M的网格地图上部署他们的炮兵部队.一个N*M的地图由N行M列组成,地图的每一格可能是山地(用“H” 表示),也可能是平原(用“P”表示),如下图.在每一格平原地形上最多 ...
- 炮兵阵地 /// 状压DP oj26314
题目大意: 炮兵阵地 设置炮兵的位置 其上两位 下两位 左两位 右两位 不能同时设置炮兵 这题是 corn fields玉米地 的升级版 可以先看下这题的注释 更详细些 第一种方法是网上大多数题解的解 ...
随机推荐
- linux 打开文件数too many open files解决方法
出现这句提示的原因是程序打开的文件/socket连接数量超过系统设定值.查看每个用户最大允许打开的文件数量ulimit -a 其中 open files (-n) 1024 表示每个用户最大允许打开的 ...
- Zookeeper协调分布式节点demo
多台服务器和客户端通过第三方组件Zookeeper管理 public class DistributedServer { private static final String connectStri ...
- docker selinux-enabled作用
一.现象 在docker中有一个运行选项是selinux-enabled.这个选项的作用是啥? 简而言之,它提供了对docker容器中进程的selinux的控制支持.下面举例说明. 首先按照官方文档的 ...
- python tkinter GUI绘制,以及点击更新显示图片
tkinter 绘制GUI简单明了,制作一些简单的GUI足够,目前遇到的一个问题是不能同时排列显示多幅图片(目前没找到同时显示解决方法), 退而求其次,改成增加一个update按钮,每次点下按钮自动更 ...
- Java程序员新手老手都离不开八大开发工具
以下这8个工具,从代码构建到错误挤压,覆盖Java开发的全域.学习这些工具可以帮助你改善代码质量,成为一个更高效的Java开发人员.Java这个大世界中正在不断涌现新的工具.实用程序和库.如果你的首选 ...
- 实战c++中的vector系列--vector的遍历(stl算法、vector迭代器(不要在循环中推断不等于end())、operator[])
遍历一个vector容器有非常多种方法.使用起来也是仁者见仁. 通过索引遍历: for (i = 0; i<v.size(); i++) { cout << v[i] << ...
- TCP/IP详解 卷一(第三章 IP:网际协议)
IP是TCP/IP协议族中最为核心的协议.所有的TCP.UDP.ICMP及IGMP数据都以IP数据报格式传输. IP提供不可靠.无连接的数据报传送服务. 1.不可靠:就是它不能保证IP数据报能成功地到 ...
- 一种调用dll的巧妙方法
直接上代码,后面说应用场景 新建一个项目,引入需要调用的dll,如下 class Program { [DllImport( "soft.dll" )] static extern ...
- c# emit 动态实现接口
using System; using System.Linq; using System.Reflection; using System.Reflection.Emit; namespace Te ...
- Maven的安装以及在IDEA中的配置
Maven的安装 之前的一篇博客中已经写到过了Maven的安装.这里就只给出链接了. http://www.cnblogs.com/tuhooo/p/5905569.html 版本虽然不同,但是安装的 ...