题目链接:https://www.rqnoj.cn/problem/328

题意:

  司令部的将军们打算在N*M的网格地图上部署他们的炮兵部队。

  一个N*M的地图由N行M列组成(N≤100,M≤10),地图的每一格可能是山地(用'H' 表示),也可能是平原(用'P'表示),如下图。

  在每一格平原地形上最多可以布置一支炮兵部队(山地上不能够部署炮兵部队);

  一支炮兵部队在地图上的攻击范围如图中黑色区域所示:

    

  如果在地图中的灰色所标识的平原上部署一支炮兵部队,则图中的黑色的网格表示它能够攻击到的区域:沿横向左右各两格,沿纵向上下各两格。

  图上其它白色网格均攻击不到。从图上可见炮兵的攻击范围不受地形的影响。

  现在,将军们规划如何部署炮兵部队,在防止误伤的前提下(保证任何两支炮兵部队之间不能互相攻击,即任何一支炮兵部队都不在其他支炮兵部队的攻击范围内)。

  问在整个地图区域内最多能够摆放多少我军的炮兵部队。

题解:

  表示状态:

    m<=10,所以对每一行进行压缩。

    dp[i][state1][state2] = max num of cannons(炮兵数量)

    i:该摆第i行

    state1:上上一行的状态(i-2)

    state2:上一行的状态(i-1)

  找出答案:

    max dp[n][state1][state2]

    枚举state1,state2.

  如何转移:

    now: dp[i][state1][state2]

    枚举第i行填的方案为nex。

    如果nex符合第i行的地形,并且与state1,state2均没有交集,则:

    dp[i+1][state2][nex] = max dp[i][state1][state2] + sum[nex]

    (sum[nex]为方案nex中的炮兵个数。)

  边界条件:

    dp[0][0][0] = 0

    others = -1

    (第0行的上两行方案均设为0,因为不填适用于任何地形)

  优化:

    (1)预处理field数组。

      field[i]为第i行的地形。每一位上1位山地,0为平原。

      判断nex是否适用于第i行时,只需判断是否有 state & field == 0

    (2)预处理在一行上的合法方案。

      在每一行上,两个炮兵之间最少相距2格。利用此性质dfs然后存起来就好。

    (3)dp数组的下标state不再直接表示01状态,而是换成s代表当前方案在dfs数组中的索引。

      省空间啊。。。

    (4)预处理出每一种行上合法方案的sum值,存起来。

      预处理时,最好用lowbit直接找出1,而不用逐位枚举。

AC Code:

 // optimizations:
// scheme: put == 1, blank == 0
// field: mountain == 1, plain == 0
// legal: scheme & field == 0
// preprocess: legal scheme for rows
//
// state expression:
// dp[i][state1][state2] = max num of cannons
// i: considering ith row
// state1: top row
// state2: bottom row
//
// find the answer:
// max dp[n][state1][state2]
//
// transferring:
// now: dp[i][state1][state2]
// if nex & field[i] == 0
// dp[i+1][state2][nex] = max dp[i][state1][state2] + sum[nex]
//
// boundary:
// dp[0][0][0] = 0
// others = -1
#include <iostream>
#include <stdio.h>
#include <string.h>
#include <stack>
#define MAX_N 105
#define MAX_M 15
#define MAX_S 65 using namespace std; int n,m;
int cnt;
int ans;
int sum[MAX_S];
int field[MAX_N];
int legal_state[MAX_S];
int dp[MAX_N][MAX_S][MAX_S]; void dfs(int col,int state)
{
if(col>=m)
{
legal_state[cnt++]=state;
return;
}
dfs(col+,state);
dfs(col+,state|(<<col));
} int cal_sum(int state)
{
int lowbit=state&-state;
int cnt=;
while(lowbit)
{
cnt++;
state^=lowbit;
lowbit=state&-state;
}
return cnt;
} void read()
{
memset(field,,sizeof(field));
cin>>n>>m;
char c;
for(int i=;i<n;i++)
{
for(int j=;j<m;j++)
{
cin>>c;
field[i]<<=;
if(c=='H') field[i]|=;
}
}
} void solve()
{
cnt=;
dfs(,);
for(int i=;i<cnt;i++)
{
sum[i]=cal_sum(legal_state[i]);
}
memset(dp,-,sizeof(dp));
dp[][][]=;
for(int i=;i<n;i++)
{
for(int s1=;s1<cnt;s1++)
{
for(int s2=;s2<cnt;s2++)
{
int state1=legal_state[s1];
int state2=legal_state[s2];
if(dp[i][s1][s2]!=- && !(state1&state2))
{
for(int s3=;s3<cnt;s3++)
{
int nex=legal_state[s3];
if(!(nex&field[i]) && !(nex&state1) && !(nex&state2))
{
dp[i+][s2][s3]=max(dp[i+][s2][s3],dp[i][s1][s2]+sum[s3]);
}
}
}
}
}
}
ans=;
for(int s1=;s1<cnt;s1++)
{
for(int s2=;s2<cnt;s2++)
{
ans=max(ans,dp[n][s1][s2]);
}
}
} void print()
{
cout<<ans<<endl;
} int main()
{
read();
solve();
print();
}

RQNOJ 328 炮兵阵地:状压dp的更多相关文章

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

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

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

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

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

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

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

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

  5. luogu 2704 炮兵阵地 状压dp

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

  6. POJ 1185 炮兵阵地 状压dp

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

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

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

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

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

  9. 炮兵阵地 /// 状压DP oj26314

    题目大意: 炮兵阵地 设置炮兵的位置 其上两位 下两位 左两位 右两位 不能同时设置炮兵 这题是 corn fields玉米地 的升级版 可以先看下这题的注释 更详细些 第一种方法是网上大多数题解的解 ...

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

    题目: P2704 [NOI2001]炮兵阵地 解析: 和互不侵犯一样 就是多了一格 用\(f[i][j][k]\)表示第i行,上一行状态为\(j\),上上行状态为\(k\)的最多的可以放的炮兵 发现 ...

随机推荐

  1. Cocos2d-x中Vector&lt;T&gt;容器以及实例介绍

    Vector<T> 是Cocos2d-x 3.x推出的列表容器,因此它所能容纳的是Ref及子类所创建的对象指针,其中的T是模板,表示能够放入到容器中的类型,在Cocos2d-x 3.x中T ...

  2. js向后台传递对象

    js: }; $.ajax({ url: "/.../...", type: "POST", async: false, data: JSON.stringif ...

  3. msgsnd的一个小问题

    今天写了一个System V消息队列的小样例.定义了一个例如以下的结构体: #define MSG_SIZE 8192 struct request { long mtype; int client_ ...

  4. GitHub使用问题(遇到一个记一个)

    1.如何创建文件夹: 如图,Create new files,点击后,若需要创建文件,输入文件名即可,但如果创建的是文件夹,需要在文件夹名后 加一个 '/'斜杠,然后就变成文件夹了

  5. ExtJs4学习(二):Dom操作

    如今主流的JS框架要数ExtJs和JQuery应用的比較广泛.JQuery属于轻量级的,一般做站点应用比較常见.可见块头小的优势. ExtJs比較庞大,它除了对主要的JS语法和HTML DOM操作方式 ...

  6. 跟我一起写 Makefile(一)[转]

    原文链接 http://bbs.chinaunix.net/thread-408225-1-1.html(出处: http://bbs.chinaunix.net/) 陈皓 概述—— 什么是makef ...

  7. php 字符串内容是数组格式 转换成数组

    一个简单的应用.. 例, $str    =    "array( 'USD'=>'1', 'GBP'=>'0.6494', 'EUR'=>'0.7668' ,'JPY'= ...

  8. 深入Asyncio(一)入门介绍

    介绍 Asyncio试图解决什么问题? 对于IO负载,有且仅有两个理由使用基于asyncio的并发而不是基于多线程的并发: 1. Asyncio为抢占式多任务(线程)提供了一个更安全的替代方案,避免了 ...

  9. 一种流量成本节省60%以上的手机直播微信直播H5直播幼儿园直播方案

    前言 近几年视频直播可以说是非常火热,EasyDarwin也非常受开发者的欢迎,不仅仅是主播火了,而且各种商业直播也火了起来:会场直播.宴会直播.讲座直播.景区直播.后厨直播.课堂直播.幼儿园直播等等 ...

  10. python循环导入的解决方案

    解决循环import的方法主要有几种:   1.延迟导入(lazy import)   即把import语句写在方法或函数里面,将它的作用域限制在局部. 这种方法的缺点就是会有性能问题.     2. ...