P2704 [NOI2001] 炮兵阵地 题解

P2704

  • 题目
    • 题目描述
    • 输入格式
    • 输出格式
    • 样例 #1
      • 样例输入 #1
      • 样例输出 #1
      • 提示
  • 解题思路
    • 分析
    • Code
      • 更多方法


题目

原题链接

题目描述

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

一个 N × M N\times M N×M 的地图由 N N N M M M 列组成,地图的每一格可能是山地(用 H \texttt{H} H 表示),也可能是平原(用 P \texttt{P} P 表示),如下图。

在每一格平原地形上最多可以布置一支炮兵部队(山地上不能够部署炮兵部队);一支炮兵部队在地图上的攻击范围如图中黑色区域所示:

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

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

现在,将军们规划如何部署炮兵部队,在防止误伤的前提下(保证任何两支炮兵部队之间不能互相攻击,即任何一支炮兵部队都不在其他支炮兵部队的攻击范围内),在整个地图区域内最多能够摆放多少我军的炮兵部队。

输入格式

第一行包含两个由空格分割开的正整数,分别表示 N N N M M M

接下来的 N N N 行,每一行含有连续的 M M M 个字符,按顺序表示地图中每一行的数据。

输出格式

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

样例 #1

样例输入 #1

5 4
PHPP
PPHH
PPPP
PHPP
PHHP

样例输出 #1

6

提示

对于 100 % 100\% 100% 的数据, 1 ≤ N ≤ 100 1 \leq N\le 100 1N100 1 ≤ M ≤ 10 1 \leq M\le 10 1M10,保证字符仅包含 PH

解题思路

分析

由于每一行都受上两行的状态的影响,所以我们需要开三维数组。
我们令 1 1 1 表示此地放了阵地, 0 0 0 则表示没有。
定义 f [ i ] [ s ] [ s 1 ] f[i][s][s1] f[i][s][s1] 表示当前在第 i i i 行,本行状态为 s s s,上一行状态为 s 1 s1 s1 时最多有几个阵地。
则循环枚举,有:

f[i][s][s1]=max(f[i][s][s1],f[i-1][s1][s2]+a[s]);

其中 s 2 s2 s2 为上两行状态, a [ s ] a[s] a[s] 表示当状态为 s s s 时放了几个阵地。
但必须满足:

  1. 状态 s , s 1 , s 2 s,s1,s2 s,s1,s2 均是合法状态。
  2. s 2 s2 s2 不影响 s , s 1 s,s1 s,s1 s 1 s1 s1 不影响 s s s

所以纯暴力的代码就已经出来了,不过:
时间复杂度: O ( n m 2 8 m ) O(nm^28^m) O(nm28m)
空间复杂度: O ( n 4 m ) O(n 4^m) O(n4m)
好吧,根本不会炸

考虑优化:

  1. 对于每个 a [ i ] a[i] a[i],可以预处理解决。
  2. 对于每个状态 s s s 本身是否合法可以预处理。
  3. 对于每个状态 s s s 的下一行有几个合法状态可以预处理。
  4. 对于每个地形,可以用二进制来表示, 1 1 1 为山地, 0 0 0 为平地。判断是 & 一下就行了。
  5. 第一维的 i i i 可以滚动掉。

对于 1
很简单:

inline int ga(int x)
{int cnt=0;while(x>0){if(x&1)cnt++;x>>=1;}return cnt;
}

对于 2

inline bool check(int x)
{return !(((x>>2)&x)|(x&(x>>1)));
}for(int i=0;i<(1<<m);i++)if(check(i))s.push_back(i);

即每两个 1 1 1 之间至少隔两个 0 0 0
对于 3

	int len=s.size();for(int i=0;i<len;i++)for(int j=0;j<len;j++)if(!(s[i]&s[j]))Q[s[i]].push_back(s[j]);

对于 4
同样很简单:

for(int i=1;i<=n;i++){for(int j=1;j<=m;j++){cin>>c;g[i]=(g[i]<<1)+(c=='H');}}

Code

 #include<bits/stdc++.h>
#define int long long
#define IOS ios::sync_with_stdio(false),cin.tie(NULL),cout.tie(NULL);
using namespace std;
const int N=1<<10;
vector<int>s;
vector<int>Q[N];
int n,m,f[101][N][N],g[200],a[N];
char c;
inline bool check(int x)
{return !(((x>>2)&x)|(x&(x>>1)));
}
inline int ga(int x)
{int cnt=0;while(x>0){if(x&1)cnt++;x>>=1;}return cnt;
}
signed main()
{IOS;cin>>n>>m;for(int i=1;i<=n;i++){for(int j=1;j<=m;j++){cin>>c;g[i]=(g[i]<<1)+(c=='H');}}for(int i=0;i<(1<<m);i++)if(check(i))s.push_back(i),a[i]=ga(i);int len=s.size();for(int i=0;i<len;i++)for(int j=0;j<len;j++)if(!(s[i]&s[j]))Q[s[i]].push_back(s[j]);int Y=0;for(int i=1;i<=n;i++){for(int j=0;j<len;j++){	Y++;int S=s[j];if(!(g[i]&S)){int L=Q[S].size();for(int k=0;k<L;k++){int S1=Q[S][k];int Le=Q[S1].size();for(int u=0;u<Le;u++){int S2=Q[S1][u];if((S&S2)||(S1&g[i-1])||(S2&g[i-2]))continue;f[i][S][S1]=max(f[i][S][S1],f[i-1][S1][S2]+a[S]);}}}}}int ans=0;for(int i=0;i<len;i++)for(int j=0;j<Q[s[i]].size();j++)ans=max(ans,f[n][s[i]][Q[s[i]][j]]);cout<<ans;return 0;
}

空间复杂度: O ( 4 m ) O(4^m) O(4m)
时间复杂度: O ( n 8 m ) O(n8^m) O(n8m)

而由于优化过后了,时间复杂度远远跑不满,实际复杂度约为: O ( 60 × [ 20 , 40 ] 2 × n ) O(60\times [20,40]^2 \times n) O(60×[20,40]2×n)

更多方法

更多方法

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/173772.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

BP算法推导

例子1&#xff1a; 解&#xff1a; 首先有如下结论&#xff1a; E k 1 2 ∑ k 1 n ( z k − f k ( x k ) ) 2 \color{green} E_k \frac{1}{2}\sum_{k1}^{n} ( z_k - f_k(x_k))^2 Ek​21​k1∑n​(zk​−fk​(xk​))2 z k ∑ i 1 , j 1 3 , 2 y i v i j z_k \sum_{i1,j …

C#常见的设计模式-创建型模式

引言 在软件开发过程中&#xff0c;设计模式是一种被广泛采用的思想和实践&#xff0c;可以提供一种标准化的解决方案&#xff0c;以解决特定问题。设计模式分为三种类型&#xff1a;创建型模式、结构型模式和行为型模式。本篇文章将重点介绍C#中常见的创建型模式。 目录 引言…

数据结构与算法编程题26

计算二叉树深度 #define _CRT_SECURE_NO_WARNINGS#include <iostream> using namespace std;typedef char ElemType; #define ERROR 0 #define OK 1 #define Maxsize 100 #define STR_SIZE 1024typedef struct BiTNode {ElemType data;BiTNode* lchild, * rchild; }BiTNo…

传音荣获2023首届全国人工智能应用场景创新挑战赛“智能家居专项赛”三等奖

近日&#xff0c;中国人工智能学会与科技部新一代人工智能发展研究中心联合举办2023首届全国人工智能应用场景创新挑战赛&#xff08;2023 1st China’s Innovation Challenge on Artificial Intelligence Application Scene&#xff0c;以下简称CICAS 2023)&#xff0c;吸引了…

Java基础之面向对象

Java基础之面向对象 一、面向对象1.1、面向对象概念1.2、面向对象的三大特征 二、类与对象2.1、构造方法2.2、创建对象2.3、实例化2.3.1、什么是实例化2.3.2、实例化的过程2.3.3、实例化的用途 三、面向对象三大特征3.1、封装3.1.1、封装概念理解3.1.2、封装目标3.1.3、封装实现…

供配电系统智能化监控

供配电系统智能化监控是指利用先进的监测技术、自动化控制技术、计算机网络技术等&#xff0c;对供配电系统进行实时、全方位的监测和控制&#xff0c;以实现供配电系统的安全、稳定、高效运行。 供配电系统智能化监控的主要功能包括&#xff1a; 实时数据采集&#xff1a;通过…

Cpython编译后再使用Pyinstaller打包

一、Cpython Python是一门解释型语言&#xff0c;当我们想让其他人运行我们的代码时&#xff0c;如果直接将.py源代码发送给他人&#xff0c;那么源代码将没有任何安全性可言&#xff0c;也就是任何一个人都可以打开源代码一看究竟&#xff0c;任何人都可以随意修改源代码。 …

编程中常见的技术难题有哪些?

编程中常见的技术难题有哪些&#xff1f; 编程中常见的技术难题有如同一道道难题&#xff0c;比如bug像隐藏的恶魔&#xff0c;让程序员们捉摸不透&#xff1b;性能优化就像是调整汽车引擎&#xff0c;需要精准的调校&#xff1b;还有就是跨平台兼容性&#xff0c;就像是翻译不…

Linux(CentOS7.5):新增硬盘分区纪实

一、服务器概述 1、既有一块系统硬盘&#xff0c;新增一块100G硬盘。 2、要求&#xff0c;将新插入硬盘分为&#xff1a;20G、30G、50G。 二、操作步骤 1、确认新硬盘是否插入成功&#xff1a; fdisk -l# 红色框出来的&#xff0c;为识别出来的新硬盘信息 # 黄色框出来的&#…

09_Map集合

Map 集合 认识 Map 集合 Map 集合称为双列集合&#xff0c;格式 { key1 value1, key2 value2, key3 value3e}&#xff0c;它也被称为"键值对集合" Map 集合的所有键是不允许重复的&#xff0c;但值可以重复&#xff0c;键和值是一一对应的&#xff0c;每一个键只…

idea创建不了spring2.X版本,无法使用JDK8,最低支持JDK17 , 如何用idea创建spring2.X版本,使用JDK8解决方案

&#x1f9f8;欢迎来到dream_ready的博客&#xff0c;&#x1f4dc;相信您对博主首页也很感兴趣o (ˉ▽ˉ&#xff1b;) &#x1f4dc;jdk17安装全方位手把手安装教程 / 已有jdk8了&#xff0c;安装JDK17后如何配置环境变量 / 多个不同版本的JDK&#xff0c;如何配置环境变量&a…

【开源】基于Vue.js的高校大学生创业管理系统的设计和实现

项目编号&#xff1a; S 027 &#xff0c;文末获取源码。 \color{red}{项目编号&#xff1a;S027&#xff0c;文末获取源码。} 项目编号&#xff1a;S027&#xff0c;文末获取源码。 目录 一、摘要1.1 项目介绍1.2 项目录屏 二、功能模块2.1 系统公告模块2.2 创业项目模块2.3 创…

直播预告 | AR眼镜在现代医疗中究竟有哪些妙用?11.28晚八点虹科直播间为您揭晓!

直播预告 | AR眼镜在现代医疗中究竟有哪些妙用&#xff1f;11.28晚八点虹科直播间为您揭晓&#xff01; 什么是AR眼镜&#xff1f; AR眼镜&#xff0c;即增强现实眼镜&#xff0c;是一种结合虚拟信息与真实世界的创新医疗工具。 通过集成高科技传感器和实时数据处理技术&…

【Python】使用globals()函数成功解决tkinter多个新窗口问题

我在近期的一个项目&#xff08;tkinter复刻记事本&#xff09;上遇到一个很有意思的问题&#xff1a;如何在创建多个新窗口后&#xff0c;每个窗口还能独立运行&#xff1f;当时我尝试几种方法&#xff0c;奈何实力不足&#xff0c;于是便下定结论非使用线程不可&#xff0c;至…

测试用例的设计思路

接到提测单后要做的事情&#xff1a; 确认提测单内包含的文件、URL地址可以访问确认需求 (迭代目标、用户故事、用户愿望、问题反馈等)确认回归测试范围、更新测试范围、新增测试范围编写测试点思维导图&#xff0c;过程中有问题及时进行沟通与迭代相关人员约一个时间, 开内部…

通付盾Web3专题 | SharkTeam:起底朝鲜APT组织Lazarus Group,攻击手法及洗钱模式

国家级APT&#xff08;Advanced Persistent Threat&#xff0c;高级持续性威胁&#xff09;组织是有国家背景支持的顶尖黑客团伙&#xff0c;专门针对特定目标进行长期的持续性网络攻击。朝鲜APT组织Lazarus Group就是非常活跃的一个APT团伙&#xff0c;其攻击目的主要以窃取资…

JavaScript WebApi(二) 详解

监听事件 介绍 事件监听是一种用于在特定条件下执行代码的编程技术。在Web开发中&#xff0c;事件监听器可以用于捕获和响应用户与页面交互的各种操作&#xff0c;如点击、滚动、输入等。 事件监听的基本原理是&#xff0c;通过在特定元素上注册事件监听器&#xff0c;当事件…

【深度学习】神经网络训练过程中不收敛或者训练失败的原因

在面对模型不收敛的时候&#xff0c;首先要保证训练的次数够多。在训练过程中&#xff0c;loss并不是一直在下降&#xff0c;准确率一直在提升的&#xff0c;会有一些震荡存在。只要总体趋势是在收敛就行。若训练次数够多&#xff08;一般上千次&#xff0c;上万次&#xff0c;…

免费SSL证书有效期只有90天?太短?

随着网络安全问题日益受到重视&#xff0c;SSL证书成为了网站安全的必需品。然而&#xff0c;在许多情况下&#xff0c;免费提供的SSL证书往往只有90天的有效期&#xff0c;这种期限对于很多用户来说显得过于短暂。 首先&#xff0c;我们要理解为什么 SSL 证书的有效期设定为90…

基于单片机DHT11湿度测量与控制-CO2-光照报警系统程序和仿真

一、系统方案 1、本设计采用这51单片机作为主控器。 2、DHT11温湿度、CO2、光照强度送到液晶1602显示。 3、按键设置报警值。 4、蜂鸣器报警。 二、硬件设计 原理图如下&#xff1a; 三、单片机软件设计 1、首先是系统初始化 //初始化LCD*********************************…