P9905 [COCI 2023/2024 #1] AN2DL 【矩阵区间最大值】

文章目录

    • 题目大意
      • 1.输入格式
      • 2.输出格式
      • 3.数据范围与约定
    • 思路
      • 维护每一行区间
      • 维护每一列区间
      • 维护区间最大值
      • code↓
    • 完结撒花( ̄▽ ̄) /

题目大意

给定 n , m , r , s n,m,r,s n,m,r,s 和一个 n × m n\times m n×m 的整数矩阵 A A A,求它每个 r × s r\times s r×s 的子矩阵的元素最大值。

1.输入格式

第一行两个整数 n , m n,m n,m 表示矩阵的高和宽。

接下来 n n n 行每行 m m m 个整数,表示矩阵 A A A

最后一行两个整数 r , s r,s r,s

2.输出格式

n − r + 1 n-r+1 nr+1 行,每行 m − s + 1 m-s+1 ms+1 个数,第 i i i j j j 列的数表示以 ( i , j ) (i,j) (i,j) 为左上角的 r × s r\times s r×s 的子矩阵元素的最大值,即 max ⁡ i ≤ x ≤ i + r − 1 , j ≤ y ≤ j + s − 1 A x , y \max\limits_{i\leq x\leq i+r-1,j\leq y\leq j+s-1}A_{x,y} ixi+r1,jyj+s1maxAx,y

3.数据范围与约定

对于 100 % 100\% 100% 的数据, 1 ≤ n , m ≤ 4000 1\leq n,m\leq 4000 1n,m4000 ∣ A i , j ∣ ≤ 10000 \lvert A_{i,j}\rvert\leq 10000 Ai,j10000 1 ≤ r ≤ n 1\leq r\leq n 1rn 1 ≤ s ≤ m 1\leq s\leq m 1sm

子任务特殊性质分值
1 1 1 n , m ≤ 40 n,m\leq 40 n,m40 r = n r=n r=n s = m s=m s=m 12 12 12
2 2 2 n , m ≤ 40 n,m\leq 40 n,m40 17 17 17
3 3 3 n , m ≤ 1000 n,m\leq 1000 n,m1000 25 25 25
4 4 4无特殊性质 56 56 56

思路

这道题我们可以用单调队列来维护矩阵最大值和矩阵最小值

先维护每一排的区间 ( i ∼ i + r ) (i\sim i+r) (ii+r)的最大值

可以用code来输出区间的起始值↓

#include <bits/stdc++.h>
using namespace std;
int main(){
	int n,m,r,s;
	cin>>n>>m>>r>>s;
	for(int i=1;i<=n-r+1;i++){
		for(int j=1;j<=m-s+1;j++){
			cout<<"("<<i<<","<<j<<")"<<endl;
		}
	}
	return 0;
}

运行结果如下↓

在这里插入图片描述
此图中的 ( x , y ) (x,y) (x,y)就是区间起始点

例如 ( 2 , 3 ) (2,3) (2,3)就如下图↓
在这里插入图片描述
图中用红色方框框起来的便是 ( 2 , 3 ) (2,3) (2,3)表示的整个区间

其中 × \times ×标起来的地方便是区间起始点,也就是 ( 2 , 3 ) (2,3) (2,3)

维护每一行区间

我们要维护每一列的最大值,而每一列区间起始点可用代码输出↓

#include <bits/stdc++.h>
using namespace std;
int main(){
	int n,m,r,s;
	cin>>n>>m>>r>>s;
	for(int i=1;i<=n;i++){
		for(int j=1;j<=m-s+1;j++){
			cout<<"("<<i<<","<<j<<")"<<endl;
		}
	}
	return 0;
}

运行结果如下↓

在这里插入图片描述
在如下图可用 × \times ×表示,下图用 × \times ×依次表示了 ( 1 , 1 ) (1,1) (1,1) ( 2 , 2 ) (2,2) (2,2) ( 3 , 3 ) (3,3) (3,3) ( 4 , 1 ) (4,1) (4,1)横排区间所在的起始点
在这里插入图片描述

维护每一列区间

我们要维护每一列区间的最大值,而每一列区间起始点可用代码输出↓

#include <bits/stdc++.h>
using namespace std;
int main(){
	int n,m,r,s;
	cin>>n>>m>>r>>s;
	for(int i=1;i<=n;i++){
		for(int j=1;j<=m-s+1;j++){
			cout<<"("<<i<<","<<j<<")"<<endl;
		}
	}
	return 0;
}

在如下图可用 × \times ×表示,下图用 × \times ×依次表示了 ( 1 , 1 ) (1,1) (1,1) ( 2 , 2 ) (2,2) (2,2) ( 3 , 3 ) (3,3) (3,3) ( 1 , 4 ) (1,4) (1,4) ( 2 , 5 ) (2,5) (2,5)竖列区间所在的起始点
在这里插入图片描述

维护区间最大值

整个区间的最大值都会集中在code↓

	for(int i=1;i<=n-r+1;i++)
		for(int j=1;j<=m-s+1;j++)

如下图↓
在这里插入图片描述
图中的 × \times ×表示的就是区间最大值所在的位置

只需要将纵列单调队列横排单调队列进行合并即可,其中ans[]数组便是用来存储区间最大值的数组

code↓

#include <bits/stdc++.h>
using namespace std;
int n,m,a[4005][4005],ans[4005][4005],r,s,b[4005][4005];//ans是答案数组,n行,m列,求r行,s列的矩阵最大值
int main(){
	cin>>n>>m;
	ios::sync_with_stdio(false);
	cin.tie(0),cout.tie(0);
	for(int i=1;i<=n;i++){
		for(int j=1;j<=m;j++){
			cin>>a[i][j];//输入初始矩阵
		}
	}
	cin>>r>>s;//输入需要求的矩阵的行数和列数
	for(int i=1;i<=n;i++){
		deque<int> mx;//定义一个单调队列,用来存储横排的序号
		for(int j=1;j<=m;j++){
			while(!mx.empty()&&a[i][mx.back()]<=a[i][j]) mx.pop_back();//判断是否非空,满足单调性
			mx.push_back(j);//将j给压入mx这个队列
			while(!mx.empty()&&mx.front()<=j-s) mx.pop_front();//队列的头不在这个区间内,将它弹出
			if(j>=s) b[i][j-s+1]=a[i][mx.front()]; //求出a[i][j]~a[i][j+s]这个区间中的最大值
		}
	}
	for(int j=1;j<=m-s+1;j++){//区间的竖列起点是1~(m-s+1)
		deque<int> mn;//定义一个单调队列,用来存储竖列的序号
		for(int i=1;i<=n;i++){
			while(!mn.empty()&&b[mn.back()][j]<=b[i][j]) mn.pop_back();//判断是否非空,用竖列的序号去满足单调性
			mn.push_back(i);//将j压入mn这个序列
			while(!mn.empty()&&mn.front()<=i-r) mn.pop_front();//求出竖列中的区间最大值
			if(i>=r) ans[i-r+1][j]=b[mn.front()][j];//这里是求出数列中的区间最大值,ans数组来进行存储,最后输出就行
		}		
	}
	for(int i=1;i<=n-r+1;i++){//横列开始的起点
		for(int j=1;j<=m-s+1;j++){//竖列开始的起点
			cout<<ans[i][j]<<' ';//输出答案
		}
		cout<<endl;
	}
	return 0;
}

完结撒花( ̄▽ ̄) /

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Spring的Bean的生命周期 | 有图有案例

Spring的Bean的生命周期 Spring的Bean的生命周期整体过程实例化初始化服务销毁循环依赖问题 完整生命周期演示 Spring的Bean的生命周期 Spring Bean的生命周期&#xff1a;从Bean的实例化之后&#xff0c;通过反射创建出对象之后&#xff0c;到Bean称为一个完整的对象&#xf…

数据中台:数字中国战略关键技术实施

这里写目录标题 前言为何要建设数据中台数据中台建设痛点数据中台学习资料聚焦前沿&#xff0c;方法论体系更新与时俱进&#xff0c;紧跟时代热点深入6大行业&#xff0c;提炼实践精华大咖推荐&#xff0c;数字化转型必备案头书 前言 在数字中国这一国家战略的牵引下&#xff0…

SpringBoot-yaml语法

1.概念 在Springboot的项目中&#xff0c;配置文件有以下几种格式&#xff1a; Application.propertiesApplication.yamlApplication.yml 其中官方推荐我们使用yaml的格式(因为能表示的数据类型很多样) 2.基本语法 # yaml形式的配置文件# 普通的key-value&#xff08;分号之后…

【AI Agent系列】【MetaGPT多智能体学习】4. 基于MetaGPT的Team组件开发你的第一个智能体团队

本系列文章跟随《MetaGPT多智能体课程》&#xff08;https://github.com/datawhalechina/hugging-multi-agent&#xff09;&#xff0c;深入理解并实践多智能体系统的开发。 本文为该课程的第四章&#xff08;多智能体开发&#xff09;的第二篇笔记。主要是对MetaGPT中Team组件…

“智联招聘崩了?” 2024春招持续升温,求职者的热情“暴涨”到服务器都承受不住了!

2 月 28 日&#xff0c;”智联招聘崩了“登上微博热搜。有网友感叹&#xff0c;现在找工作太难了&#xff0c;发现有这么多人在竞争更焦虑了。 对此智联招聘回应称&#xff0c;由于求职流量新高&#xff0c;服务器过载&#xff0c;造成了短暂停用&#xff0c;但目前服务已恢复正…

设计模式(十三)抽象工厂模式

请直接看原文:设计模式&#xff08;十三&#xff09;抽象工厂模式_抽象工厂模式告诉我们,要针对接口而不是实现进行设计。( )-CSDN博客 -------------------------------------------------------------------------------------------------------------------------------- …

基于Mahout实现K-Means聚类

需求分析 需要对数据集进行预处理&#xff0c;选择合适的特征进行聚类分析&#xff0c;确定聚类的数量和初始中心点&#xff0c;调用Mahout提供的K-Means算法进行聚类计算&#xff0c;评估聚类结果的准确性和稳定性。同时&#xff0c;需要对Mahout的使用和参数调优进行深入学习…

解析社交媒体二维码生成:连接世界,拓展社交网络

随着社交媒体的普及和二维码技术的发展&#xff0c;社交媒体二维码作为一种新型的社交工具&#xff0c;逐渐受到人们的关注和喜爱。本文将探讨社交媒体二维码的定义、应用场景以及优势所在。 1. 什么是社交媒体二维码? 社交媒体二维码是将个人或企业在社交媒体平台上的信息整…

JavaSE-09(Java IO精华总结)

Java IO 简单做个总结&#xff1a; 1 .InputStream/OutputStream 字节流的抽象类。2 .Reader/Writer 字符流的抽象类。3 .FileInputStream/FileOutputStream 节点流&#xff1a;以字节为单位直接操作“文件”。4 .ByteArrayInputStream/ByteArrayOutputStream 节点流&#xff…

现在如何才能开通微信公众号留言功能?

为什么公众号没有留言功能&#xff1f;2018年2月12日之后直到现在&#xff0c;新注册公众号的运营者会发现一个问题&#xff1a;无论是个人还是企业的公众号&#xff0c;在后台都找不到留言功能了。这对公众号来说绝对是一个极差的体验&#xff0c;少了一个这么重要的功能&…

4. 编写app组件

1. 代码 main.ts // 引入createApp用于创建应用 import {createApp} from "vue"// 引入App根组件 import App from ./App.vue createApp(App).mount(#app) App.vue <!-- vue文件可以写三种标签1. template标签&#xff0c;写html结构2. script 脚本标签&…

Python实现向量自回归移动平均模型(VARMA算法)项目实战

说明&#xff1a;这是一个机器学习实战项目&#xff08;附带数据代码文档视频讲解&#xff09;&#xff0c;如需数据代码文档视频讲解可以直接到文章最后获取。 1.项目背景 向量自回归移动平均模型&#xff08;Vector Autoregressive Moving Average, VARMA&#xff09;是一种…

GEE:使用ReLu激活函数对单波段图像进行变换(以NDVI为例)

作者:CSDN @ _养乐多_ 本文将介绍在 Google Earth Engine (GEE)平台上,对任意单波段影像进行 ReLu 变换的代码。并以对 NDVI 影像像素值的变换为例。 文章目录 一、ReLu激活函数1.1 什么是 ReLu 激活函数1.2 用到遥感图像上有什么用?二、代码链接三、完整代码一、ReLu激活…

电脑端微信无法打开公众号/小程序?

生气了。怎么动不动电脑端微信就打不开公众号/小程序&#xff0c;每次都要搞好久 第一步&#xff1a;打开控制面板、网络和Internet&#xff0c;选中Internet选项。 第二步&#xff1a;选中连接选项&#xff0c;并打开下方的局域网设置。 第三步&#xff1a;取消 为LAN使用代理…

横幅条(自定义控件)的编写

常可以看见ViewPager翻页视图下有几个圆点&#xff0c;并随着图片变化而变化。 我们称之为横幅条&#xff0c;横幅条自定义控件的编写有两种&#xff1a;(1)使用Paint与Canvas绘制&#xff1b;(2)使用RadioButton组合。 第一种编写方法的优势是可以显示滑动过程中的位置&#…

大模型之SORA技术学习

文章目录 sora的技术原理文字生成视频过程sora的技术优势量大质优的视频预训练库算力多&#xff0c;采样步骤多&#xff0c;更精细。GPT解释力更强&#xff0c;提示词(Prompt&#xff09;表现更好 使用场景参考 Sora改变AI认知方式&#xff0c;开启走向【世界模拟器】的史诗级的…

Arduino应用开发——使用GUI-Guider制作LVGL UI并导入ESP32运行

Arduino应用开发——使用GUI-Guider制作LVGL UI并导入ESP32运行 目录 Arduino应用开发——使用GUI-Guider制作LVGL UI并导入ESP32运行前言1 使用GUI-Guider设计UI1.1 创建工程1.2 设计UI 2 ESP工程导入UI2.1 移植LVGL2.2 移植UI文件2.3 调用UI文件2.4 烧录测试 结束语 前言 GU…

(UE4升级UE5)Selected Level Actor节点升级到UE5

本问所用工具为&#xff1a; UE5 UE4 插件AssetDeveTool包含&#xff1a;快速选择功能自动化批量LOD功能自动化批量展UV功能自动化批量减面功能自动化批量修改查找替换材质功能批量重命名工具碰撞器修改工具资源整理工具支持4.26 - 5.3版本https://mbd.pub/o/bread/mbd-ZZubkp…

Manomotion 实现AR手势互动-解决手势无效的问题

之前就玩过 Manomotion &#xff0c;现在有新需求&#xff0c;重新接入发现不能用了&#xff0c;不管什么办法&#xff0c;都识别不了手势&#xff0c;我记得当初是直接调用就可以的。 经过研究发现&#xff0c;新版本SDK改了写法。下边就写一下新版本的调用&#xff0c;并且实…

Windows如何安装docker-desktop

下载 docker-desktop设置环境安装wsl可能遇到的错误 下载 docker-desktop 下载官网&#xff1a;https://www.docker.com/products/docker-desktop/ 设置环境 如果没有Hyper-V选项的,按照以下步骤 添加一个文件Hyper-V.bat 添加以下内容,并双击运行后重启电脑 pushd "%~…