炮兵阵地

司令部的将军们打算在 N×M 的网格地图上部署他们的炮兵部队。一个 N×M的地图由 N 行 M 列组成,地图的每一格可能是山地(用 H 表示),也可能是平原(用 P表示),如下图。在每一格平原地形上最多可以布置一支炮兵部队(山地上不能够部署炮兵部队);一支炮兵部队在地图上的攻击范围如图中黑色区域所示:

如果在地图中的灰色所标识的平原上部署一支炮兵部队,则图中的黑色的网格表示它能够攻击到的区域:沿横向左右各两格,沿纵向上下各两格。图上其它白色网格均攻击不到。从图上可见炮兵的攻击范围不受地形的影响。
现在,将军们规划如何部署炮兵部队,在防止误伤的前提下(保证任何两支炮兵部队之间不能互相攻击,即任何一支炮兵部队都不在其他支炮兵部队的攻击范围内),在整个地图区域内最多能够摆放多少我军的炮兵部队。

输入格式

第一行包含两个由空格分割开的正整数,分别表示 N 和 M;
接下来的 N 行,每一行含有连续的 M 个字符(P 或者 H),中间没有空格。按顺序表示地图中每一行的数据。

输出格式

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

样例

样例输入

5 4
PHPP
PPHH
PPPP
PHPP
PHHP

样例输出

6

数据范围与提示

N≤100,M≤10。

 ------------------------------------------------------------------------------------------------------------------------------------------------------------------------
状态压缩动态规划
f[i][s][ss]:表示填充到第i行且第i行的状态为st[s]第i-1行的状态为st[ss]时对多能填充多少个
f[i][s][ss]=max(f[i-1][ss][sss])
------------------------------------------------------------------------------------------------------------------------------------------------------------------------
 1 #include<bits/stdc++.h>
2 using namespace std;
3 int n,m;
4 long long f[105][70][70];
5 vector<int> st[105];
6 vector<int> jc[105];
7 int stt[105];
8 int js;
9 char ts[13];
10 void getst()
11 {
12 st[0].push_back(0);jc[0].push_back(0);
13 for(int i=1;i<=n;++i)
14 {
15 for(int j=0;j<(1<<m);++j)
16 {
17 if((j&(j<<1))==0 && (j&(j<<2))==0 && (j&(j>>1))==0 && (j&(j>>2))==0 && (j&stt[i])==j)
18 {
19 st[i].push_back(j);
20 int tp=0;
21 for(int x=0;x<m;++x)if(j&(1<<x))tp++;
22 jc[i].push_back(tp);
23 }
24 }
25 }
26 }
27 void dp()
28 {
29 for(int i=0;i<st[1].size();++i)f[1][i][0]=jc[1][i];
30 for(int i=2;i<=n;++i)
31 {
32 for(int s=0;s<st[i].size();++s)
33 {
34 for(int ss=0;ss<st[i-1].size();++ss)
35 {
36 if((st[i][s]&st[i-1][ss])==0)
37 for(int sss=0;sss<st[i-2].size();++sss)
38 {
39 if((st[i][s]&st[i-2][sss])==0 && (st[i-1][ss] &st[i-2][sss])==0)
40 f[i][s][ss]=max(f[i][s][ss],f[i-1][ss][sss]+jc[i][s]);
41 }
42 }
43
44 }
45 }
46 }
47 int main()
48 {
49 scanf("%d%d",&n,&m);
50 for(int i=1;i<=n;++i)
51 {
52 scanf("%s",ts);
53 for(int tp,j=0;j<m;++j)
54 {
55 stt[i]=(stt[i]<<1)|(ts[j]=='P');
56 }
57 }
58 getst();
59 dp();
60 long long ans=0;
61 for(int s=0;s<st[n].size();++s)
62 {
63 for(int ss=0;ss<st[n-1].size();++ss)
64 ans=max(ans,f[n][s][ss]);
65 }
66 cout<<ans;
67 return 0;
68 }
 

loj10173的更多相关文章

随机推荐

  1. 工具类输出当前ApplicationContext所有被装配的类

    package org.springblade.desk.utils; import org.springframework.beans.BeansException; import org.spri ...

  2. 学习Promise异步编程

    JavaScript引擎建立在单线程事件循环的概念上.单线程( Single-threaded )意味着同一时刻只能执行一段代码.所以引擎无须留意那些"可能"运行的代码.代码会被放 ...

  3. hadoop3.2+Centos7+5个节点主从模式配置

    准备工作: hadoop3.2.0+jdk1.8+centos7+zookeeper3.4.5 以上是我搭建集群使用的基础包 一.环境准备 master1 master2 slave1 slave2 ...

  4. 【Go】四舍五入在go语言中为何如此困难

    四舍五入是一个非常常见的功能,在流行语言标准库中往往存在 Round 的功能,它最少支持常用的 Round half up 算法. 而在 Go 语言中这似乎成为了难题,在 stackoverflow ...

  5. Linux 时间同步 04 ntp时间同步

    Linux 时间同步 04 ntp时间同步 目录 Linux 时间同步 04 ntp时间同步 安装ntp 配置与外部时间服务器进行时间同步的客户端主机 配置其他客户端与以上客户端主机时间同步 验证查看 ...

  6. 入门oj 6492: 小B的询问

    Description 小B有一个序列,包含N个1~K之间的整数.他一共有M个询问,每个询问给定一个区间[L..R],求Sigma(c(i)^2)的值,其中i的值从1到K,其中c(i)表示数字i在[L ...

  7. java中装箱,拆箱

    (1)包装类 java中不能定义基本数据类型的对象,但是提供了基本数据类型的包装类来解决这一问题,例如int类型的包装类Integer, 包装类的常用方法有Integer.ParseInt()方法类将 ...

  8. redis数据类型:String

    redis数据类型:String 花开堪折直需折,莫待无花空折枝 实际应用中有多种多样的场景,比如: 秒杀 新闻热点 在线人数 session.token管理 各个场景需要的数据类型各不相同:常见的数 ...

  9. label_form

    表单: action "URL" 如果为空,则本form接收 指定接收方 disabled 指定该标签是否可用 method "net" "http& ...

  10. VsCode配置Go语言插件

    前言 宇宙第一IDE对于笔记本来说还是太过沉重了 VsCode虽然差了点但是胜在插件多且够轻量 VsCode的安装/汉化参考我之前的博客 https://www.cnblogs.com/chnmig/ ...